扫描类下的注解,哪些包下的类需要使用IOC注解
<!-- Spring 的注解开发:组件扫描(类上注解: 可以直接使用属性注入的注解) -->
<context:component-scan base-package="com.itheima.spring.demo1"/>
@Component(value="userDao")
public class UserDaoImpl implements UserDao {
@Override
public void sayHello() {
System.out.println("Hello Spring Annotation...");
}
}
相当于在xml中用bean标签配置类
Spring 中提供@Component 的三个衍生注解:(功能目前来讲是一致的)
* @Controller :WEB 层
* @Service :业务层
* @Repository :持久层
这三个注解是为了让标注类本身的用途清晰,Spring 在后续版本会对其增强
@Value :用于注入普通类型.
@Autowired :自动装配:
* 默认按类型进行装配.
* 按名称注入:
* @Qualifier:强制使用名称注入.
@Resource 相当于:
* @Autowired 和@Qualifier 一起使用.
@Scope:
- singleton:单例
- prototype:多例
@PostConstruct :相当于 init-method @PreDestroy :相当于 destroy-method
XML 和注解:
- XML :结构清晰.
- 注解 :开发方便.(属性注入.) 实际开发中还有一种 XML 和注解整合开发:
注意:注解方式的这个类是自己提供,但是xml的话可以适用所有类
Bean 有 XML 配置.但是使用的属性使用注解注入.
在没有扫描下,使用属性注入的注解,需要加上这个配置
<context:annotation-config>
AOP是指面向切面编程,是oop的延续,可以为业务逻辑的各个部分进行隔离,解决oop开发遇到的问题,使得业务间的耦合性降低
可以对程序,或者通俗的说是对类的方法进行增强,在不修改源代码的条件下
AOP可以进行权限校验,日志记录,性能监控,事务控制
底层利用了代理机制
Spring 的 AOP 的底层用到两种代理机制:
- JDK 的动态代理 :针对实现了接口的类产生代理.
- Cglib 的动态代理 :针对没有实现接口的类产生代理. 应用的是底层的字节码增强的技术 生成当前类 的子类对象.
1)JDK 动态代理增强一个类中方法
public class MyJDKProxy implements InvocationHandler {
private UserDao userDao;
public MyJDKProxy(UserDao userDao) {
this.userDao = userDao;
}
// 编写工具方法:生成代理:
public UserDao createProxy(){
UserDao userDaoProxy = (UserDao)
Proxy.newProxyInstance(userDao.getClass().getClassLoader(),
userDao.getClass().getInterfaces(), this);
return userDaoProxy;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable
{
if("save".equals(method.getName())){
System.out.println("权限校验================");
}
return method.invoke(userDao, args);
}
}
2)Cglib 动态代理增强一个类中的方法
public class MyCglibProxy implements MethodInterceptor{
private CustomerDao customerDao;
public MyCglibProxy(CustomerDao customerDao){
this.customerDao = customerDao;
}
// 生成代理的方法:
public CustomerDao createProxy(){
// 创建 Cglib 的核心类:
Enhancer enhancer = new Enhancer();
// 设置父类:
enhancer.setSuperclass(CustomerDao.class);
// 设置回调:
enhancer.setCallback(this);
// 生成代理:
CustomerDao customerDaoProxy = (CustomerDao) enhancer.create();
return customerDaoProxy;
}
@Override
public Object intercept(Object proxy, Method method, Object[] args, MethodProxy
methodProxy) throws Throwable {
if("delete".equals(method.getName())){
Object obj = methodProxy.invokeSuper(proxy, args);
System.out.println("日志记录================");
return obj;
}
return methodProxy.invokeSuper(proxy, args);
}
}
可以被拦截到的点,在 spring 中,这些点指的是方法,因为 spring 只支持方法类型的连接点。
真正被拦截到的点,就是我们真正需要去增强的方法
就是为切入点增强功能的那些个方法,比如权限校验方法等.通知分为前置通知,后置 通知,异常通知,最终通知,环绕通知(切面要完成的功能)
是切入点和通知(引介)的结合
引介是一种特殊的通知在不修改类代码的前提下, Introduction 可以在运行期为类动态地添加一些方法或 Field.,是类层面的
:代理的目标对象
是指把增强应用到目标对象来创建新的代理对象的过程,spring 采用动态代理织入,
一个类被 AOP 织入增强后,就产生一个结果代理类
引入Spring-test的jar包,不需要再创建工厂,直接测试
类注解:
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
依赖注入:
@Resource(name="orderDao")
将增强的方法放进来,比如权限校验等
<!-- 配置切面类 -->
<bean id="myAspectXml" class="cn.itcast.spring.demo3.MyAspectXml"></bean>
<!-- 进行 aop 的配置 -->
<aop:config>
<!-- 配置切入点表达式:哪些类的哪些方法需要进行增强 -->
<aop:pointcut expression="execution(*
cn.itcast.spring.demo3.OrderDao.save(..))" id="pointcut1"/>
<!-- 配置切面 -->
<aop:aspect ref="myAspectXml">
<aop:before method="before" pointcut-ref="pointcut1"/>
</aop:aspect>
</aop:config>
<aop:config>
<!-- 表达式配置哪些类的哪些方法需要增强 点点代表任意参数-->
<aop:pointcut expression="execution(* com.canghe.demo1.PruductDaoImpl.save(..))" id="pointcut1"/>
<aop:pointcut expression="execution(* com.canghe.demo1.PruductDaoImpl.delete(..))" id="pointcut2"/>
<aop:pointcut expression="execution(* com.canghe.demo1.PruductDaoImpl.update(..))" id="pointcut3"/>
<aop:pointcut expression="execution(* com.canghe.demo1.PruductDaoImpl.find(..))" id="pointcut4"/>
<!-- 配置切面 -->
<aop:aspect ref="myAspect">
<!-- 配置前置通知 -->
<aop:before method="checkPri" pointcut-ref="pointcut1"/>
<!-- 配置后置通知 -->
<aop:after-returning method="weiteLog" pointcut-ref="pointcut2" returning="result"/>
<!-- 配置环绕通知 -->
<aop:around method="around" pointcut-ref="pointcut3"/>
<!-- 配置异常抛出通知 -->
<aop:after-throwing method="afterThrowing" pointcut-ref="pointcut4" throwing="ex" />
<!-- 配最终异知 -->
<aop:after method="after" pointcut-ref="pointcut4" />
</aop:aspect>
</aop:config>
每种通知均可获得切入点信息
如权限校验
在目标方法执行之前执行
<aop:before method="checkPri" pointcut-ref="pointcut1"/>
如:日志记录
在目标方法执行之后执行
获得方法的返回值
<!-- 配置后置通知 -->
<aop:after-returning method="weiteLog" pointcut-ref="pointcut2" returning="result"/>
然后将result作为参数传入切面中的通知
如:性能监控
在目标方法执行前和执行后执行
可以阻止目标方法的执行
在通知类方法要有Object返回值,设置如下:
/**
* 环绕通知——性能监控
* @throws Throwable
*/
public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("环绕通知前==============");
Object object = joinPoint.proceed();
System.out.println("环绕通知后==============");
return object;
}
在目标方法执行出现 异常的时候 执行
可得到异常信息,
<!-- 配置异常抛出通知 -->
<aop:after-throwing method="afterThrowing" pointcut-ref="pointcut4" throwing="ex" />
/**
* 异常抛出通知
*/
public void afterThrowing(Throwable ex) {
System.out.println("异常抛出通知============="+ex.getMessage());
}
无论目标方法是否出现异常 最终通知都会 执行
<!-- 配最终异知 -->
<aop:after method="after" pointcut-ref="pointcut4" />
execution(表达式)
表达式:
[方法访问修饰符] 方法返回值 包名.类名.方法名(方法的参数)
public * cn.itcast.spring.dao.*.*(..)
* cn.itcast.spring.dao.*.*(..)
* cn.itcast.spring.dao.UserDao+.*(..)
* cn.itcast.spring.dao..*.*(..)
鞥切面扩展程序,想要增强某一个方法,来一个切面然后配置一下就好了,不需要修改源代码