(二)Spring AOP源码-3.创建AOP代理

整理了下创建代理的过程。

创建代理过程
1. 获取增强器
	1.1 获取所有的beanname
	1.2 遍历所有的beanname,找出声明@AspectJ注解的类,提取增强器
	1.3 对标记为@AspectJ注解的类进行增强器的提取
		1.3.1 获取切点信息
			1.3.1.1 获取方法上的注解并封装
		1.3.2 根据切点信息生成增强器
			1.3.2.1 根据不同的注解类型封装不同的增强器,比如@Before会对应AspectJMethodBeforeAdvice。在AspectJMethodBeforeAdvice中完成了增强逻辑
	1.4 将提取结果加入缓存

2. 获取匹配的增强器
	2.1 引介增强和普通增强分开处理

3. 根据获取的增强器进行代理
	3.1 初始化proxyFactory代理工厂
		3.1.1 获取当前类中的属性
		3.1.2 添加代理接口
		3.1.3 封装advisor并添加到proxyFactory中
		3.1.4 设置要代理的类
		3.1.5 对proxyFactory的定制封装
		3.1.6 进行获取代理操作
	3.2 创建代理
		3.2.1 创建代理对象:代理工厂会选择JdkDynamicAopProxy或者CglibAopProxy,主要通过是否接口和是否配置cglib代理来选择。
			3.2.1.1 参数1:Optimize,用来控制创建的CglibAopProxy是否采用激进的优化策略
			3.2.1.2 参数2:proxytragetclass,如果为true,则创建CglibAopProxy
			3 2.1.3 hasNoUserSuppliedProxyInterfaces,是否存在代理接口
	3.3 获取代理
		3.3.1 jdk代理
			3.3.1.1 核心是InvocationHandler
				3.3.1.1.1 构造方法,将代理的对象传入
				3.3.1.1.2 invoke方法,实现了aop增强的所以逻辑
				3.3.1.1.3 getProxy方法,千篇一律,必不可少
			3.3.1.2 JdkDynamicAopProxy确实实现了InvocationHandler,看看它的invoke方法实现
				3.3.1.2.1 获取当前方法的拦截器链
				3.3.1.2.2 如果没有获取到拦截器链,则直接调用切点方法
				3.3.1.2.3 如果获取到了拦截器链,则将拦截器链封装在ReflectiveMethodInvocation,以便试用其process方法执行拦截器链,看看拦截器链的执行逻辑。ReflectiveMethodInvocation的主要职责是维护了链接调用的计数器,记录着当前调用链接的位置,以便链可以有序地执行下去
		3.3.1 cglib代理,核心是Cglib2AopProxy,完成cglib代理的类是委托给Cglib2AopProxy来实现的,在Cglib2AopProxy的getProxy方法中实现了Enhancer的创建和封装,再通过Enhancer来完成代理的创建。
			3.3.1.1 cglib对于方法的拦截是是通过自定义的拦截器(实现MethodInterceptor接口)加入callback中并在调用代理直接激活拦截器中的intercept方法来实现的。
			3.3.1.2 CGLIB实现的代理,核心逻辑在DynamicAdvisedInterceptor中的intercept方法中。
			3.3.1.3 intercept方法和jdk代理实现中的invoke方法大同小异,都是先构造拦截器链,再封装此链进行链式调用。区别在于jdk代理中直接构造ReflectiveMethodInvocation,而在cglib代理中使用CglibMethodInterceptor。CglibMethodInterceptor继承了ReflectiveMethodInvocation,但process方法并没有重写。下面总结下拦截器链的构造过程:
			3.3.1.3.1 遍历所有的增强器,将其转为Interceptor;registry.getInterceptors(advisor);
			3.3.1.3.2 将增强器转为List<MethodInterceptor>;如果是MethodInterceptor,直接加入到集合中;如果不是,使用AdvisorAdapter将增强器转为MethodInterceptor;转换完成返回MethodInterceptor数组;
		3.3.1.4 如果没有拦截器链,直接执行目标方法;
		3.3.1.5 如果有拦截器链,把需要执行的目标对象,目标方法,拦截器链等信息传入创建一个CglibMethodInvocation 对象,并调用proceed(),链式获取每一个拦截器,拦截器执行invoke方法,每一个拦截器等待下一个拦截器执行完成返回以后再来执行。
4. 给容器中返回当前组件使用cglib增强了的代理对象。以后容器中获取到的就是这个组件的代理对象,执行目标方法的时候,代理对象就会执行通知方法的流程;
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值