核心概念
AOP(Aspect Oriented Programming)面向切面编程,一种编程范式,指导开发者如何组织程序结构
- OOP(0bject 0riented Programming)面向对象编程
- 作用: 在不惊动原始设计的基础上为其进行功能增强
- AOP理念: 无入侵式/无侵入式
核心概念
- 连接点(JoinPoint): 程序执行过程中的任意位置,粒度为执行方法、抛出异常、设置变量等
- 在SpringAOP中,理解为方法的执行
- 实际被增强的方法称为连接点
- 切入点(Pointcut): 匹配连接点的式子
- 在SpringAOP中,一个切入点可以只描述一个具体方法,也可以匹配多个方法
- 一个具体方法: com.itheima.dao包下的BookDao接口中的无形参无返回值的save方法
- 匹配多个方法: 所有的save方法,所有的get开头的方法,所有以Dao结尾的接口中的任意方法,所有带有一个参数的方法
- 定义哪些方法需要被增强称为切入点
- 通知(Advice): 在切入点处执行的操作,也就是共性功能
- 在SpringAOP中,功能最终以方法的形式呈现
- 增强的具体功能称为通知
- 通知类: 定义通知的类
- 切面(Aspect):
- 描述通知与切入点的对应关系
- 绑定通知与切入点称为切面
入门案例
思路分析
案例设定: 测定接口执行效率
简化设定: 在接口执行前输出当前系统时间
开发模式: XML or 注解
思路分析
- 导入坐标(pom.xml)
- 制作连接点方法(原始操作,Dao接口与实现类)
- 制作共性功能(通知类与通知 )
- 定义切入点
- 绑定切入点与通知关系(切面)
代码实现
- 导入aop相关坐标
说明: spring-context坐标依赖spring-aop坐标, 所以不用单独导入了
- 定义dao接口与实现类
- 定义通知类,制作通知
- 把待增强的功能抽取出来
- package com.itheima.aop
- 定义切入点
- 说明: 切入点定义依托一个不具有实际意义的方法进行,即无参数,无返回值,方法体无实际逻辑
- 绑定切入点与通知关系,并指定通知添加到原始连接点的具体执行位置
- @Pointcut 定义切入点 (指定哪些方法需要被增强)
- @Before("pt()") 绑定通知与切入点 (指定执行时机和增强方法)
- 定义通知类受Spring容器管理,并定义当前类为切面类
- 开启Spring对AOP注解驱动支持
工作流程解析
- Spring容器启动
- 读取所有切面配置中的切入点
- 初始化bean,判定bean对应的类中的方法是否匹配到任意切入点
- 匹配失败,创建对象
- 匹配成功,创建原始对象(目标对象)的代理对象
- 获取bean执行方法
- 获取bean,调用方法并执行,完成操作
- 获取的bean是代理对象时,根据代理对象的运行模式运行原始方法与增强的内容,完成操作
- SpringAOP本质: 通过代理模式进行功能增强
- 目标对象(Target): 原始功能去掉共性功能对应的类产生的对象,这种对象是无法直接完成最终工作的
- 代理(Proxy): 目标对象无法直接完成工作,需要对其进行功能回填,通过原始对象的代理对象实现
切入点表达式
切入点: 要进行增强的方法
切入点表达式: 要进行增强的方法的描述方式
- 描述方式一: 执行com.itheima.dao包下的BookDao接口中的无参数update方法
- execution(void com.itheima.dao.BookDao.update())
- 描述方式二: 执行com.itheima.dao.impl包下的BookDaoImpl类中的无参数update方法
- execution(void com.itheima.dao.impl.BookDaoImpl.update())
切入点表达式标准格式:
动作关键字(访问修饰符 返回值 包名.类/接口名.方法名(参数) 异常名)
- 动作关键字: 描述切入点的行为动作,例如execution表示执行到指定切入点
- 访问修饰符: public,private等,可以省略
- 返回值
- 包名
- 类/接口名
- 方法名
- 参数
- 异常名: 方法定义中抛出指定异常,可以省略
可以使用通配符描述切入点,快速描述
- *: 单个独立的任意符号,可以独立出现,也可以作为前缀或者后缀的匹配符出现
匹配com.itheima包下的任意包中的UserService类或接口中所有find开头的带有一个参数的方法
- ..: 多个连续的任意符号,可以独立出现,常用于简化包名与参数的书写
匹配com包下的任意包中的UserService类或接口中所有名称为findByld的方法
- +: 专用于匹配子类类型
书写技巧
- 所有代码按照标准规范开发,否则以下技巧全部失效
- 描述切入点通常描述接口,而不描述实现类, 因为实现了可能会变
- 访问控制修饰符针对接口开发均采用public描述(可省略访问控制修饰符描述)
- 返回值类型对于增删改类使用精准类型加速匹配,对于查询类使用*通配快速描述
- 包名书写尽量不使用..匹配,效率过低,常用*做单个包描述匹配,或精准匹配
- 接口名/类名书写名称与模块相关的采用*匹配,例如UserService书写成*Service,绑定业务层接口名
- 方法名书写以动词进行精准匹配,名词采用*匹配,例如getByld书写成getBy*,selectAll书写成selectAll
- 参数规则较为复杂,根据业务方法灵活调整
- 通常不使用异常作为匹配规则
通知类型
AOP通知描述了抽取的共性功能,根据共性功能抽取的位置不同,最终运行代码时要将其加入到合理的位置
AOP通知共分为5种类型
- 前置通知
- 后置通知
- 环绕通知(重点)
- 返回后通知(了解 )
- 抛出异常后通知(了解)
前置通知
- 名称:@Before
- 类型:方法注解
- 位置:通知方法定义上方
- 作用:设置当前通知方法与切入点之间的绑定关系,当前通知方法在原始切入点方法前运行
- 范例:
- 相关属性: value(默认):切入点方法名,格式为类名.方法名()
后置通知
- 名称: @After
- 类型方法注解
- 位置: 通知方法定义上方
- 作用: 设置当前通知方法与切入点之间的绑定关系,当前通知方法在原始切入点方法后运行
- 范例
- 相关属性: value(默认):切入点方法名,格式为类名.方法名(
环绕通知(重点)
- 名称: @Around(重点,常用)
- 类型: 方法注解
- 位置: 通知方法定义上方
- 作用: 设置当前通知方法与切入点之间的绑定关系,当前通知方法在原始切入点方法前后运行
- 范例:
- @Around注意事项
- 环绕通知必须依赖形参ProceedingJoinPoint才能实现对原始方法的调用,进而实现原始方法调用前后同时添加通知
- 通知中如果未使用ProceedingJoinPoint对原始方法进行调用将跳过原始方法的执行
- 对原始方法的调用可以不接收返回值,通知方法设置成void即可,如果接收返回值,必须设定为Obiect类
- 原始方法的返回值如果是void类型,通知方法的返回值类型可以设置成void,也可以设置成Obiect
- 由于无法预知原始方法运行后是否会抛出异常,因此环绕通知方法必须抛出Throwable对象
返回后通知(了解 )
- 名称: @AfterReturning(了解)
- 类型: 方法注解
- 位置: 通知方法定义上方
- 作用: 设置当前通知方法与切入点之间的绑定关系,当前通知方法在原始切入点方法正常执行完毕后运行
- 范例:
- 相关属性:value(默认):切入点方法名,格式为类名.方法名()
- 如果目标方法执行中出现异常, 该通知不会触发
抛出异常后通知(了解)
- 名称: @AfterThrowing(了解)
- 类型: 方法注解
- 位置: 通知方法定义上方
- 作用: 设置当前通知方法与切入点之间的绑定关系,当前通知方法在原始切入点方法运行抛出异常后执行
- 范例
- 相关属性: value(默认):切入点方法名,格式为类名.方法名()
练习案例
测量业务层接口执行效率
- 需求: 任意业务层接口执行均可显示其执行效率(执行时长)
- 分析
- 业务功能: 业务层接口执行前后分别记录时间,求差值得到执行效率
- 通知类型选择前后均可以增强的类型--环绕通知
- 补充说明
- 当前测试的接口执行效率仅仅是一个理论值,并不是一次完整的执行过程
代码实现
- 引入aop依赖坐标
- 添加注解, 让Spring识别AOP通知类
- 定义AOP通知类
- 编写具体的增强逻辑, 统计方法执行时长
- 执行结果
通知获取数据
AOP通知获取数据
- 获取切入点方法的参数
- JoinPoint: 适用于前置、后置、返回后、抛出异常后通知
- ProceedJointPoint: 适用于环绕通知
- 获取切入点方法返回值
- 返回后通知
- 环绕通知
- 获取切入点方法运行异常信息
- 抛出异常后通知
- 环绕通知
AOP通知获取参数数据
- JoinPoint对象描述了连接点方法的运行状态,可以获取到原始方法的调用参数
- ProceedJointPoint是JoinPoint的子类
AOP通知获取返回值数据
- 抛出异常后通知 可以获取切入点方法中出现的异常信息,使用形参可以接收对应的异常对象
- 环绕通知中 可以手工书写对原始方法的调用得到的结果即为原始方法的返回值
AOP通知获取异常数据(了解 )
- 抛出异常后通知可以获取切入点方法中出现的异常信息,使用形参可以接收对应的异常对象
- 抛出异常后通知可以获取切入点方法运行的异常信息,使用形参可以接收运行时抛出的异常对象
练习案例
百度网盘分享链接输入密码数据错误兼容性处理
- 需求: 对百度网盘分享链接输入密码时尾部多输入的空格做兼容处理
- 分析
- 在业务方法执行之前对所有的输入参数进行格式处理--trim()
- 使用处理后的参数调用原始方法--环绕通知中存在对原始方法的调用
代码实现
- 引入aop坐标
- 开启aop注解开发
- 定义通知类, 其中定义切入点和通知
- 在通知方法中完成业务, 核心逻辑就是拿到通知数据, 对数据进行操作, 把修改过的参数传给目标方法, 类似数据篡改的效果