核心概念
advice:通知,就是你想要实现的功能,比如日志、事务
JoinPoint:切点,理解为需要切入的方法
Pointcut:切入点,理解为需要切入的方法的位置
Aspect(Advisor): Aspect=advice+Pointcut,切面=干什么+在哪干
AOP源码解析
AOP切面方法调用流程分析
1、解析注解获取所有的通知advisor列表
2、根据通知列表构造增强链chain,对原始方法进行增强。这一过程通过调用时执行ReflectiveMethodInvocation.proceed方法实现。
3、DynamicAdvisedInterceptor是enhance的interceptor。负责构造MethodInvocation并调用proceed方法。把对原始方法的调用委托给methodInvocation。
4、MethodInvocation里面是aop的interceptor。aop的interceptor接口如下
public interface MethodInterceptor extends Interceptor {
Object invoke(MethodInvocation invocation) throws Throwable;
}
常见的aop的Interceptor有:MethodBeforeAdviceInterceptor,AfterAdviceInterceptor,AspectJAroundAdvice,AspectJAfterReturningAdvice,AspectJAfterThrowingAdvice
5、通过method.invoke实现切面方法的调用
判断一个类是否需要增强以及增强哪些切面
AbstractAutoProxyCreator有如下重要属性:
advisedBeans:保存哪些bean需要增强。对应的bean的方法中满足advises增强太条件的那些bean
advisorsCache:通知缓存,key为通知名(logAspecct),value为该通知对应的Advisor列表
1、AbstractAutoProxyCreator.postProcessAfterInitialization会检查每个bean通过wrapIfNecessary判断是否需要生成代理类;
2、findCandidateAdvisors找出所有的advised:找出所有含有@Aspect注解的bean;过滤出该类中非@Pointcut注解的方法,每个方法都封装成一个Advisor,并放入advisorsCache;
3、过滤出可匹配当前bean的advsior,调用AopUtils.findAdvisorsThatCanApply.
类增强的代码实现
参考CglibAopPoxy.getProxy方法
ProxyFactory构造时候传入TargetSource
1、从TargetSource获取待增强的目标类
2、检查类增强限制:非final类,非final方法
3、Enhancer的Interceptor:SpringProxy(标识是否是spring aop产生的代理类)和Advised。
4、Enhancer的Callbacks设置,该数组共有7个元素:StaticUnadvisedInterceptor。
5、设置Enhancer的ProxyCallbackFilter。
DynamicAdvisedInterceptor
StaticunadvisedInterceptor
SerializableNoOp:对应final,不做增强
EqualsInterceptor
DynamicAdvisedInterceptor方法
每个待增强的方法都会操作一遍
1、调用getInterceptorsAndDynamicInterceptionAdvice,构造拦截的chain。
2、aop各个advise的调用在ReflectiveMethodInvocation.proceed
3、ReflectiveMethodInvocation封装了chain列表,和index指示器,会递归调用全部的interceptors。
4、调用的第一个aopinterceptor是ExposeInvocationInterceptor,用于保存MethodInvocation上下文,该interceptor优先级最高,值最小,最先执行
Objenesis Vs class.newInstance
从以上代码可以发现class构造器需要参数,而Objenesis可以绕过去, Objenesis主要应用场景:
- 序列化,远程调用和持久化 -对象需要实例化并存储为到一个特殊的状态,而没有调用代码。
- 代理,AOP库和Mock对象 -类可以被子类继承而子类不用担心父类的构造器。
- 容器框架 -对象可以以非标准的方式被动态实例化。
详细参考:https://blog.csdn.net/codershamo/article/details/52015206
生成代理对象AopProxy
有两种方式CglibAopProxy和JdkDynamicAopProxy
ApplicationContext.refresh->registerBeanPostProcessors
aop源码分析:https://zhuanlan.zhihu.com/p/35135042
jdk动态代理
InvocationHandler = {}
People proxy = (People)Proxy.newProxyInstance(handler.getClass().getClassLoader(), people.getClass().getInterfaces(), handler);
cglib增强原理
cglib详解:https://blog.csdn.net/zghwaicsdn/article/details/50957474
cglib用法样例
public class ServiceCgInterceptor implements MethodInterceptor {
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("before");
Object ret = methodProxy.invokeSuper(o, objects);//调用原始
System.out.println("after");
return ret;
}
public static void main(String[] args) {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(ServiceCg.class);
enhancer.setCallback(new ServiceCgInterceptor());
ServiceCg proxy = (ServiceCg) enhancer.create();
proxy.say1();
}
}
如果增强一个类需要用到多个Interceptor,实现如下:
enhancer.setCallbacks(cbarray);
enhancer.setCallbackFilter(callbackFilter); //CallbackFilter返回对应callback的index