spring三种代理模式
1.静态代理
- 目标类需要实现一个目标类接口
- 需要自己写一个代理类用来融合目标类方法和切面类通知
2.JDK动态代理
- 目标类需要实现一个目标类接口
- 代理对象是目标接口的一个子类
- 需要一个拦截器,实现InvocationHandler接口,重写invoke方法,融合目标类方法和切面类通知
- 拦截器提供有参构造函数用来注入属性
- 测试类将目标类,切面类都传入拦截器对象
- 测试类中使用Proxy.newProxyInstance(loader, interfaces, h),JDK会自动生成一个代理类
- 代理类调用目标方法会先走拦截器的invoke方法,invoke方法里面的method.invoke(target,args)才是真正调用目标方法
- 不需要目标类接口就可以产生代理对象
- 代理对象是目标类的一个子类
- 需要一个拦截器,实现MethodInterceptor接口,重写intercept方法,融合目标类方法和切面类通知
- 拦截器提供有参构造函数用来注入属性
- 拦截器,提供createProxy方法,里面是固定写法
- 测试类调用拦截器的createProxy方法即可使用cglib包生成一个代理对象
spring框架实现动态代理
- 目标类实现接口,则sping框架用JDK创建代理类
getBean获得的是PersonDao类型 - 目标类没有实现接口,则sping框架用Cglib包创建代理类
getBean获得的是PersonDaoImpl类型
- 先将目标类和切面类加载到Spring作用域中
- 使用aop配置将目标类和切面类柔和到一起
- 并通过切入点表达式配置和切面配置告诉切面类的通知融合到目标类的哪个目标方法中
- 目标类加载对象注解@Component(“personDao”)
- 事务类加载对象注解@Component(“m”)
- 事务类切面注解@Aspect
- 事务类通知注解
前置:@Before(“execution (* com.jp.aop.PersonDaoImpl.(…))")
后置:@AfterReturning(value="execution ( com.jp.aop.PersonDaoImpl.(…))",returning=“ret”)
异常:@AfterThrowing(value="execution ( com.jp.aop.PersonDaoImpl.(…))",throwing=“xx”)
最终:@After("execution ( com.jp.aop.PersonDaoImpl.(…))")
环绕:@Around("execution ( com.jp.aop.PersonDaoImpl.*(…))”)