文章目录
- 1、AOP思想
- 2、AOP入门案例
- 3、AOP工作流程
- 4、AOP切入点表达式
- 5、AOP的五种通知类型
- 6、AOP通知获取数据
- 7、案例:百度网盘密码数据兼容处理
- 8、AOP总结
1、AOP思想
AOP,即Aspect Oriented Programming,面向切面编程
。是一种编程范式,指导开发者如何组织程序结构。在不惊动原始设计的基础上为其进行功能增强,实现无侵入式编程。
回顾:
OOP(面向对象编程)
Object Oriented Programming
AOP思想举例:
如下图右侧,我有增删改查四个方法,我想让它们的方法体被执行10000次,并计算耗时,但不能改变现有的程序。想实现这个效果,考虑定义一个共性方法(即左边的通知类里的method),并将待增强的方法(右)和共性方法做一个切面关联。
核心概念:
题外话:每用到AOP,脑子里就想到了小学改作文的加字符,不用擦掉原有的(不改现有代码),直接加。
2、AOP入门案例
思路分析:
- 导入坐标(pom.xml)
- 制作连接点方法(即待增强的代码)
- 制作共性功能(通知类与通知)
- 定义切入点
- 绑定切入点与通知关系
步骤1:导入切面相关依赖坐标(pom.xml)
步骤2:随便写个连接点方法(这里是你要做增强的方法)
步骤3:制作共性功能(通知类与通知)
步骤4:在通知类中定义切入点
切入点的定义需要依托一个不具有实际意义的方法进行,如上面的pt()方法,它无参数、无返回值、方法体无实际逻辑。其中:@Pointcut注解来定义切入点
步骤5:绑定切入点与通知的关系,并指定通知添加到原始连接点具体执行位置
注意,需要用注解@Component
定义通知类受Spring容器管理,再加注解@Aspect
定义当前类为切面类,最后在Spring核心配置类中使用注解@EnableAspectJAutoProxy
开启对AOP注解驱动支持
至此,在未改动原程序的基础上,update方法新增了功能–打印当前时间:
3、AOP工作流程
注意,若bean对应的类中的方法没有匹配到任意切入点,则创建出来的bean就是普通对象,无关AOP,也无关代理对象。
打印bean,由于toString被重写,看不出来区别,调用getClass方法,可以看到代理类class是com.sun.proxy.$Proxy19,整个AOP的实现就是使用了代理模式。
4、AOP切入点表达式
切入点表达式语法:
切入点
:需要进行增强的那个方法,如上面的update切入点表达式
:即要进行增强的这个方法,怎么去描述它
比如描述要增强的update,可以按照接口,也可以按照它的实现类:
切入点表达式标准格式:
---->
动作关键词(访问修饰符 返回值 包名.类/接口名 .方法名(参数)异常名)
execution(public User com.llg.service.Uservice.findById(int))
- 动作关键字:描述切入点的行为动作,execution即执行到指定切入点
- 方法修饰符:public、private等可以省略
- 返回值
- 包名
- 类/接口名
- 参数
- 异常名:方法定义中抛出指定异常,可以省略
通配符
使用通配符来描述切入点,提高效率:
- *:单个独立的任意符号,可以独立出现,也可做为前缀或后缀匹配符。
execution(public * com.llg.*.UserService.find* (*))
匹配com.llg包下得任意包中得UserService类或接口中所有find开头得带有一个参数(注意不是无参数)
、返回类型任意的public方法
- . .:多个连续的任意符号,可独立出现,常用于简化包名与参数的书写
execution(public User com..UserService.findById(..))
匹配com包下的任意包中的UserService类或接口中所有名称为findById的方法
- +:专门用于匹配子类类型
execution(* *..*Service+.*(..))
切入点表达式的书写技巧:
5、AOP的五种通知类型
AOP通知描述了抽取的共性功能,根据共性功能抽取的位置的不同,最终代码运行时要将其加入到合理的位置中,(说白了就是共性功能和待增强的原代码怎么个执行顺序)这个位置,有5中类型:
- 前置通知
- 后置通知
- 环绕通知
- 返回后通知
- 抛出异常后通知
代码实现:
各个类型的详解:
@Before:
@After
@Around
注意:pjp.proceed()是对原始操作的调用,这样就区分出来了环绕具体是怎么个执行顺序,当原始操作有返回值的时候,这里要改为Object类型,并接收proceed方法的返回值来return。(原始操作返回void,也可以使用Object类型,这时候返回的是null而已)
@AfterReturning
@AfterThrowing
案例--测量业务层接口万次执行效率
计算时间差即执行时长,需要在接口执行前后分别记录时间==>环绕通知
使用环绕通知,先实现执行时长:
通过执行签名信息,来获取所测的接口名和方法名:
6、AOP通知获取数据
获取切入点方法的参数:
获取切入点方法返回值
- 对于AfterReturning,我们使用形参来接收返回值:
- 对于Around,执行原始方法时,得到的结果即返回值:
获取切入点方法异常信息
- 对于Around,不再throw,直接try…catch即可:
- 对于AfterThrowing,使用新参接收异常对象:
7、案例:百度网盘密码数据兼容处理
实现核心:使用环绕通知对原始操作的参数进行修改!
代码实现:
业务层接口和实现类:
数据层接口和实现类:
使用AOP思想,为业务层方法,加上去空格的功能,提高兼容性和用户使用体验:
测试程序:
运行结果:
====================
8、AOP总结