零、准备知识

1)AOP相关概念:Aspect、Advice、Join point、Pointcut、Weaving、Target等。
2)相关注解:@Aspect、@Pointcut、@Before、@Around、@After、@AfterReturning、@AfterThrowing
 

一、实践目标

1)@Aspect的功能
2)@Pointcut的切面表达式
3)@Before、@Around、@After、@AfterReturning / @AfterThrowing的时序关系
4)AOP概念的重新梳理和使用

二、核心代码

MainController.java包含两个测试函数,分别是doSomething() 和 test()。
 // MainController.java
@RestController
public class MainController {
RequestMapping(value="/doSomething", method = RequestMethod.POST)
@CrossOrigin("*")
public void doSomething() {
System.out.println("This is doSomething");
test();
} @RequestMapping(value="/justTest", method = RequestMethod.POST)
@CrossOrigin("*")
public void test() { System.out.println("This is test");}
}

ExampleAop.java为AOP相关代码,定义了pointcut和多个Advice函数。

 // ExampleAop.java
@Component
@Aspect
@Order(1)
public class ExampleAop { private static final Logger LOGGER = LoggerFactory.getLogger(ExampleAop.class); // 匹配com.example.demo包及其子包下的所有类的所有方法
@Pointcut("execution(* com.example.demo..*.*(..))")
public void executeService() {
} @Before("executeService()")
public void doBeforeAdvice(JoinPoint joinPoint) throws Exception {
LOGGER.info("Before [{}]", joinPoint.getSignature().getName());
} @Around("executeService()")
public void doAroundAdvice(ProceedingJoinPoint joinPoint) throws Throwable {
LOGGER.info("Around1 [{}]", joinPoint.getSignature().getName());
joinPoint.proceed();
LOGGER.info("Around2 [{}]", joinPoint.getSignature().getName());
} @After("executeService()")
public void doAfterAdvice(JoinPoint joinPoint) throws Exception {
LOGGER.info("After [{}]", joinPoint.getSignature().getName());
} @AfterReturning("executeService()")
public void doAfterReturningAdvice(JoinPoint joinPoint) throws Exception {
LOGGER.info("AfterReturning [{}]", joinPoint.getSignature().getName());
}
}

编译并运行jar包,调用接口/doSomething。

 # 调用接口
curl localhost:8080/doSomething

到服务器观察日志。

 <!-- 后台日志 -->
com.example.demo.aop.ExampleAop : Around1 [doSomething]
com.example.demo.aop.ExampleAop : Before [doSomething]
This is doSomething
This is test
com.example.demo.aop.ExampleAop : Around2 [doSomething]
com.example.demo.aop.ExampleAop : After [doSomething]
com.example.demo.aop.ExampleAop : AfterReturning [doSomething]

三、分析与结论

1)@Aspect的功能
  在连接点的前后添加处理。在本例中,doSomething() 是连接点,而test() 不是。
  是否只有最外层的joinpoint才会被Advice插入?在后面进行简单的探讨和猜测。
 
 
2)@Pointcut的切面表达式
  ref: https://www.jianshu.com/p/fbbdebf200c9  完整表达式
  @Pointcut("execution(...)") 是Pointcut表达式,executeService() 是point签名。表达式中可以包含签名的逻辑运算。
 
  常用表达式:
 execution(public * com.example.demo.ExampleClass.*(..))  // ExampleClass的所有公有方法
execution(* com.example.demo..*.*(..)) // com.example.demo包及其子包下的所有方法
logSender() || logMessage() // 两个签名的表达式的并集
3)@Before、@Around、@After、@AfterReturning / @AfterThrowing的时序关系
  @Around1 -> @Before -> 方法 -> @Around2 -> @After -> @AfterReturning / @AfterThrowing(时序问题后面有额外讨论。)
  另外可以发现,@Around是可以影响程序本身执行的,如果不调用 joinPoint.proceed(); 就不会执行方法。其他几个都无法影响程序执行。
 
 
4)AOP概念的重新梳理和使用
  Aspect(切面):使用了@Aspect注解,如ExampleAop类。
  Advice(增强):在指定位置进行的增强操作,如方法运行时间统计、用户登录、日志记录等。由@Before、@After等注解标注,如doBeforeAdvice() 方法。
  Weaving(织入):AOP就是一种把Advice织入(即嵌入、插入)到Aspect中指定位置执行的机制。
  Join point(连接点):Advice执行的位置,也是Advice的参数,是一个具体的方法。如日志中看到的doSomething() 函数。
  Pointcut(切点):以表达式的形式表示一组join point,用于由@Pointcut注解定义Advice的作用位置。如@Pointcut("execution(* com.example.demo..*.*(..))") 代表com.example.demo包及其子包下的所有类的所有方法。
  Target(对象):被增强的对象,即包含主业务逻辑的类的对象,如ExampleAop类的实例。

四、疑问与讨论

1. 本文说执行顺序为@Around1 -> @Before -> 方法 -> @Around2 -> @After,但有的文章中说是@Before -> @Around1 -> 方法 -> @Around2 -> @After,也有说@Around1 -> @Before -> 方法 -> @After -> @Around2,哪个对?
 
  反正代码跑起来是这个顺序,那就是这个顺序喽。每个Advice都加上sleep拉开时间也没有变化。不知道是否受版本或代码自身影响。
  总之可以得到一个结论:@Before / @After 最好不要和@Around混用,执行顺序不好确定。
  时序至少总是满足:@Around1 / @Before -> 方法 -> @Around2 / @After -> @AfterReturning / @AfterThrowing
  另外找到一篇支持本文执行顺序的文章:https://blog.csdn.net/qq_32331073/article/details/80596084
 
 
2. 为何doSomething() 和 test() 都是@Pointcut中选中的作用节点,但只有doSomething() 插入了Advice,而test() 没有呢?
 
        一个猜测:从字面意思理解,@Pointcut对所有代码以表达式为规则剪一刀,一侧是所有的joinpoint,另一侧是普通代码。在joinpoint与另一侧代码间插入一层Advice的代理,另一侧的代码如果要调用joinpoint,则必须经Advice进行增强操作。而不同的joinpoint在同一侧,因此未插入Advice。
        有时间再读源码了解其中的机制。

(一个猜测)

五、Future Work

1. AOP中还有Advisor等概念,待学习。
2. 切面表达式(@Pointcut里的表达式)规则丰富,待学习。
3. Advice的插入时机,待读源码学习。

Spring入门之AOP实践:@Aspect + @Pointcut + @Before / @Around / @After的更多相关文章

  1. Spring入门4.AOP配置深入

    Spring入门4.AOP配置深入 代码下载 链接: http://pan.baidu.com/s/11mYEO 密码: x7wa 前言: 之前学习AOP中的一些概念,包括连接点.切入点(pointc ...

  2. Spring入门3.AOP编程

    Spring入门3.AOP编程 代码下载: 链接: http://pan.baidu.com/s/11mYEO 密码: x7wa 前言: 前面学习的知识是Spring在Java项目中的IoC或DJ,这 ...

  3. Spring基础系列--AOP实践

    原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9615720.html 本文目的是简单讲解下Spring AOP的使用. 推荐使用IDEA ...

  4. Spring入门之AOP篇

    听了几节IT黑马营的SPRING课程,照着例程写了一个SPRING 中AOP的例子:  一.准备工作 下载复制或配置JAR包.图方便,我将下载的SPRING包都加上去了.另外还需要aspectj的两个 ...

  5. Spring入门篇——AOP基本概念

    1.什么是AOP及实现方式 什么是AOP AOP:Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术 主要 ...

  6. spring 注解 之 AOP基于@Aspect的AOP配置

    Spring AOP面向切面编程,可以用来配置事务.做日志.权限验证.在用户请求时做一些处理等等.用@Aspect做一个切面,就可以直接实现. 1.首先定义一个切面类,加上@Component  @A ...

  7. Spring入门介绍-AOP(三)

    AOP的概念 AOP是面向切面编程的缩写,它是一种编程的新思想.对我们经常提起的oop(面对对象编程)有一定的联系. AOP和OOP的关系 AOP可以说是oop的某一方便的补充,oop侧重于对静态的属 ...

  8. Spring课程 Spring入门篇 5-3 配置切入点 pointcut

    1 解析 1.1 xml常见的配置切入点写法 2 代码演练 2.1 xml配置切入点   1 解析 1.1 xml常见的配置切入点写法 2 代码演练 2.1 xml配置切入点 xml配置: <? ...

  9. 《Java Spring框架》Spring切面(AOP)配置详解

    1.  Spring 基本概念 AOP(Aspect Oriented Programming)称为面向切面编程,在程序开发中主要用来解决一些系统层面上的问题,比如日志,事务,权限等待,Struts2 ...

随机推荐

  1. codeforce303C-Minimum Modular-剪枝,暴力

    Minimum Modular 题意:就是在一堆数字中,每一个数字对m取模不能等于这堆数字中的其他数字,同时给了K个机会可以删除一些数字.求最小的m: 思路:我一开始完全没思路,队长说的并查集什么的不 ...

  2. lightoj 1028 - Trailing Zeroes (I)(素数筛)

    We know what a base of a number is and what the properties are. For example, we use decimal number s ...

  3. 最小生成树问题---Prim算法学习

    一个具有n个节点的连通图的生成树是原图的最小连通子集,它包含了n个节点和n-1条边.若砍去任一条边,则生成树变为非连通图:若增加一条边,则在图中形成一条回路.本文所写的是一个带权的无向连通图中寻求各边 ...

  4. Disruptor中shutdown方法失效,及产生的不确定性源码分析

    版权声明:原创作品,谢绝转载!否则将追究法律责任. Disruptor框架是一个优秀的并发框架,利用RingBuffer中的预分配内存实现内存的可重复利用,降低了GC的频率. 具体关于Disrupto ...

  5. 浅谈独立特征(independent features)、潜在特征(underlying features)提取、以及它们在网络安全中的应用

    1. 关于特征提取 0x1:什么是特征提取 特征提取研究的主要问题是,如何在数据集未明确表示结果的前提下,从中提取出重要的潜在特征来.和无监督聚类一样,特征提取算法的目的不是为了预测,而是要尝试对数据 ...

  6. golang开发:类库篇(五)go测试工具goconvey的使用

    为什么要使用goconvey测试程序 goconvey 集成go test,go test 无缝接入.管理运行测试用例,而且提供了丰富的函数断言.非常友好的WEB界面,直观的查看测试结果. 如果没有g ...

  7. Hibernate,一对多,多对一

    Hibernate环境的配置 hibernate.cfg.xml的配置 <?xml version='1.0' encoding='utf-8'?> <!DOCTYPE hibern ...

  8. 生产环境:ansible自动化部署kubernetes-1.14

    概述: 本文提供ansible-playbooks用来帮助读者用ansible构建二进制kubernetes1.14, 集群包含calico.nginx-ingress.HA 提供资源有kuberne ...

  9. SqlServer 2014 还原数据库时提示:操作系统返回了错误5,,拒绝访问

    场景 在进行数据库还原时提示: System.Data.SqlError:在对”“尝试”“时,操作系统返回了错误5(拒绝访问) 实现 第一种方案是修改要还原的数据库备份文件的权限. 找到备份文件右击属 ...

  10. Delphi - 调用外部程序并阻塞到外部程序中

    Delphi 调用外部程序并阻塞到外部程序中 背景说明: 前段时间开发一个数据转换的系统,业务逻辑中说明数据需要压缩成.tar.gz格式. 我在Windows系统下采用,先生成批处理文件,然后调用Wi ...