Spring AOP源码分析

在实际项目中会碰到这样一种场景,在不改动业务情况下,记录接口执行日志。通常解决方案是使用基于AOP的Aspectj注解。那问题来了Sping AOP与Aspectj有何区别?

一 什么是AOP ?什么是Spring AOP?什么是AspectJ?

1 AOP 全称是“Aspect Oriented Programming”,即面向切面编程。主要作用是在不改变业务代码情况下,为业务添加另外的通用功能,AOP是代理模式的应用。适用于日志记录,事务管理,参数检验等场景。

常见术语:

Joinpoint(连接点)在 Spring 中,指可以被动态代理拦截目标类的方法(某个目标实例的函数)。

Pointcut(切入点)目标对象功能集合。

Advice(通知)指拦截到 Joinpoint 之后要做的事情,即对切入点增强的内容。

Target(目标)指代理的目标对象。

Weaving(植入)指把增强代码应用到目标上,生成代理对象的过程。

Proxy(代理)指生成的代理对象。

Aspect(切面)切入点和通知的结合。

2 Spring AOP是基于 AOP 编程模式的一个框架,常见的实现方式基于接口的 JDK 动态代理和基于继承的 CGLIB 动态代理。

3AspectJ是一个基于 Java 语言的 AOP 框架。

3.1AspectJ不同的开启方法

3.1.1 Java 配置启用 @AspectJ 支持

@Configuration

@EnableAspectJAutoProxy

public class AppConfig {

}

注解定义切面和增强示例

@Aspect

@Component

public class AspectTest {

// 定义切入点

@Pointcut("execution(* test.X.*(..))")

public void point() {}

// 前置通知

@Before("point()")

public void before() {

System.out.println("前置");

}

// 后置通知 始终会执行

@After("point()")

public void after() {

System.out.println("后置");

}

// 环绕通知

@Around("point()")

public Object around(ProceedingJoinPoint pjp) throws Throwable {

System.out.println("环绕前");

Object result = pjp.proceed();

System.out.println("环绕后");

return result;

}

// 后置 发生异常时不会执行

@AfterReturning("point()")

public void returning() {

System.out.println("After returning 后置");

}

// 发生异常

@AfterThrowing("point()")

public void throwing() {

System.out.println("发生异常了");

}

}

3.1.2 XML 配置启用 @AspectJ 支持

<aop:aspectj-autoproxy/>

基于 XML定义切面和通知(增强)示例

<aop:config>

    <aop:aspect id="myAspect" ref="aBean">

        <aop:pointcut id="businessService"

            expression="execution(* com.xyz.myapp.service.*.*(..))"/>

        <aop:before pointcut-ref="businessService" method="monitor"/>

    </aop:aspect>

</aop:config>

4 Spring(版本5.1.0.RELEASE) + Aspectj源码分析。以下面这种形式分析

    AnnotationConfigApplicationContext context =

          new AnnotationConfigApplicationContext(Cfg.class);

    X x=(X)context.getBean("x");

x.getY();

4.1 解析加了@componentScan属性的配置类并注册到bean定义中心。

4.2 后置处理器注册

执行函数PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors,循环执行后置处理器,当后置处理器为ConfigurationClassPostProcessor实例时,调用函数postProcessBeanDefinitionRegistry(registry)在内部执行函数loadBeanDefinitionsFromRegistrars(configClass.getImportBeanDefinitionRegistrars());

4.3 AspectJAutoProxyRegistrar注册到bean定义中心

执行函数processConfigBeanDefinitions解析配置类里@componentScan注解的backpages属性值加载包路径下class文件封装到Resource实例中返回Resource数组。循环遍历Resource数组,获取加了@Componet注解或注解的注解有加@Componet,创建beandefinition封装解析到属性值,注册到bean定义中心,最后返回Set<ConfigurationClass> 实例赋值给变量configClasses。循环遍历configClasses取出实例执行函数loadBeanDefinitionsForConfigurationClass,当配置类有加@EnableAspectJAutoProxy注解时执行到函数loadBeanDefinitionsFromRegistrars时会把@EnableAspectJAutoProxy定义的注解@Import(AspectJAutoProxyRegistrar.class)创建beandefinition实例注册进beandefinition中心。

AspectJAutoProxyRegistrar注册执行调用栈

4.4 循环实例化懒加载bean

执实例DefaultListableBeanFactory的函数preInstantiateSingletons() ,加载非懒加载bean循环遍历DefaultListableBeanFactory属性beannames集合,执行dogetBean函数实例化bean,再执行函数populateBean对实例属性进行赋值,最后执行函数initializeBean。

4.5 获取加了@Apsectj类下面的增强函数,并创建代理对象

在函数initializeBean内部 循环取出后置处理器,当后置处理器AbstractAutoProxyCreator实例 执行函数postProcessAfterInitialization函数内部的函数wrapIfNecessary,调用函数findCandidateAdvisors获取不到增强,再执行this.aspectJAdvisorsBuilder.buildAspectJAdvisors()函数内循环遍历beanames集合实例,获取加了@Aspect注解的类,解析类加了增强注解函数并创建Advice实例封装解析到的值返回List集合实例,再执行findAdvisorsThatCanApply(List<Advisor> candidateAdvisors, Class<?> beanClass, String beanName)函数,校验当前类是否跟@PointCut注解属性表达式匹配,匹配上返回增强集合,否则不返回了。根据增强集合有值创建对应beanname的代理对象。

获取增强和代理对象创建调用栈1

获取增强和代理对象创建调用栈2

获取增强和代理对象创建调用栈3

4.6 切面类增强函数织入目标函数中

当通过代理对象调用目标对象函数时,执行CglibAopProxy类下面的内部类DynamicAdvisedInterceptor函数intercept,函数proceed()内部首先校验currentInterceptorIndex变量与增强器集合个数是否一致,如果一致执行目标实例函数。如果不一致currentInterceptorIndex增加1,传给增强器集合实例函数get取出实例,再执行取出的增强实例的函数invoke(this),内部去执行增强函数,一直递归调用直到最后执行目标实例函数。

执行目标函数切入增强功能调用栈

有问题下发留言!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,185评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,652评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,524评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,339评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,387评论 6 391
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,287评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,130评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,985评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,420评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,617评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,779评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,477评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,088评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,716评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,857评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,876评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,700评论 2 354

推荐阅读更多精彩内容