哈哈 终于更到这里了 🐷

今天来和小伙伴们分享这个 Spring AOP 的知识点 👇

image-20210913072321674

这里先推荐大家去阅读 Spring 的官方文档,core 文档的5,6章节, 和 AOP 相关的概念都可以这这里找到~,还有我们要了解的 Spring AOP APIs 😄

image-20210910074335232

先来介绍下这个 AOP 😄

AOP 概念

这是个老生常谈的问题呀 哈哈 熟悉的小伙伴们可以略过😄

面向切面编程 Aspect-Oriented Programming ,是对 OOP 的一种补充

使用场景有 : 日志事务权限认证缓存

Spring AOP 呢,会通过这个 代理类 来实现对方法的增强 👇

image-20210910073611972

这里有很重要的几个概念 👇(看图就好啦)

image-20210910074649844

概念关系图

通过 PointCut 去匹配符合的 JoinPoint ,再对其做 Weaving·操作,将 Advice 织入其中。

PointCutAdvice 存放在 Aspect 中。

image-20210910081828619

切点表达式图

如:executionwithinthistargetargsbean

以及匹配注解用的 @target@args@within@annotation

execution

作用: 匹配 某些连接点

如:

  • execution(public (..)) 作用: 匹配所有公共方法
  • execution( set(..)) 作用:匹配所有 set 开头的方法
  • execution( com.xyz.service.AccountService.(..)) 作用:匹配 AccountService 中的所有方法
  • execution( com.xyz.service..*(..)) 作用:匹配 service 包中的所有方法
  • execution( com.xyz.service...*(..)) 作用:匹配 service 包和它的子包中 所有的方法

within

作用:匹配 类 中的所有 连接点

如:

  • within(com.xyz.service.*) 作用:匹配 service 包中的所有方法
  • within(com.xyz.service..*) 作用:匹配 service 包和它的子包中 所有的方法

this

作用:匹配 代理类

  • this(com.xyz.service.AccountService) 作用:匹配 实现了 AccountService 接口的 代理类,即 父类是 AccountService

target

作用:匹配 被代理类

  • target(com.xyz.service.AccountService) 作用:匹配 实现了 AccountService 接口的 被代理类,即 父类是 AccountService

args

作用:匹配 方法参数

  • args(java.io.Serializable) 作用:匹配 单个方法参数,且是 Serializable 类型的

bean

作用:匹配 bean 名称,Spring 扩展 AspectJ 的 PDC 规范, 只能在 Spring AOP 中生效

@target

作用:匹配 有注解的 被代理类

  • @target(org.springframework.transaction.annotation.Transactional) 作用:匹配 @Transactional 注解的 被代理类

@args

作用:匹配 传入的参数 有某个注解

  • @args(com.xyz.security.Classified) 作用: 匹配有 @Classified 注解的参数

@within

作用:匹配 有某个注解 的 类

  • @within(org.springframework.transaction.annotation.Transactional) 作用: 匹配 有 @Transactional 注解 的 类中的所有 连接点

@annotation

作用:匹配 该注解

  • @annotation(org.springframework.transaction.annotation.Transactional) 作用: 匹配 @Transactional 注解

小结图如下 👇

image-20210910082639157

Advice 注解图

这里是常用的五个注解

注意:@After 等于 finally 语句块,必定会执行,还有 @Around 注解时,方法的第一个参数必须是 ProceedingJoinPoint ,它是 JoinPoint 的子接口

image-20210913074020835


下面的内容是本文的重点~ 😝

编程式 AOP

这个呢,是我们写 AOP插件 的基础

image-20210912074920354

这里要理解这几个概念 👇

  • Pointcut : 对应 @Pointcut
  • Advice对应 @Before,@After,@Around 等等
  • Advisor@Aspect 注解解析后的类,等于 Pointcut + Advice 🐷
  • Advised : Spring AOP 代理类都可以转换成该接口! 🐷

例子我放到 Github 上啦 ,地址放文末啦👇

Pointcut

里面有两个返回类型式 ClassFilterMethodMatcher 的方法,他们的作用如下 👇

image-20210912100228554

4ye 也是大概了解下 哈哈 主要还是用 Spring 提供的实现类,如下面例子中用到的 NameMatchMethodPointcut ,看名字就能猜个大概,就是匹配这个方法名的 😄

小例子

代码例子比较多,这里就举这个 Pointcut 看看就好啦~ (其他案例代码小伙伴们可以自行在我的 Github 仓库上获取😄)

通过对 ProxyFactroy 添加 切面 来对目标 sayService 做增强,这里的 Pointcut 只针对 setName 方法

AOP_Pointcut

效果

可以看到只增强了 setName 方法

image-20210912094544020

Advice

可以看到这里 Spring 提供了这 五种类型。除了最后一种 Introduction 类型上面没提到外,是不是还发现少了一种类型😝

image-20210912161447604

没错,就是这个 @After 注解对应的最终通知 😱

这去哪了呀 ( ̄﹏ ̄;)

在源码中寻找一番后,发现了这个 AspectJAfterAdvice 👇

image-20210912104757851

可以看到这里不仅实现了 AfterAdvice 接口,还要实现这个 MethodInterceptor 接口 ,最后真的是在 finally 语句块中实现了这个增强操作,来达到 最终通知 的效果!

所以我们需要 最终通知 效果的话,可以直接模仿它去实现就好了~

而 Introduction Advice 这个还挺特别的🐖

作用:为代理类实现新的接口

等于 @DeclareParents 注解

Advisor

作用 👇

@Aspect 注解解析后的类,等于 Pointcut + Advice

Advised

作用 👇

AOP 代理类都可以转换成该接口

为啥这些代理对象都能转换成这个 Advised 接口?

答案当然不言而喻 哈哈,咱们来源码中看看 Spring 是怎么写的😝

JDK

image-20210911231149301

Cglib

image-20210911233943416

结论

可以发现,不管是 JDK 还是 Cglib , 都会去调用同一个方法,为代理类多实现几个接口,其中一个就是 Advised 😋

关系图

SpringAOP

最后

本文就分享到这里啦🐖

重点内容在 编程式AOP 这一块~ 内功+1 😋

代码可以在这里获取 👇 可以自己多跑跑代码,会有更多问题等着你发现的!

https://github.com/Java4ye/springboot-demo-4ye/tree/main/springboot-aop

思维导图 👇

http://processon.com/chart_image/6134d6cb7d9c081c753b5d54.png

ps:这次分享的时候居然出现了 Gateway Timeout 异常 ,如果等不及官方修复的话,可以在网盘获取

链接https://pan.baidu.com/s/1KVWw0DaYCKrISIKJgqqyuA

提取码 需在公众号后台回复 bdwp

image-20210913074314344

插件的部分代码我也上传啦,就差这画画图写写文啦 哈哈 (这两天应该就安排上了)😆

喜欢的话可以 关注星标 下公众号 Java4ye 支持下 4ye 呀😝,这样就可以第一时间收到更文消息啦🐷

我是4ye 咱们下期应该……很快再见!! 😆