- AOP(Aspect Oriented Programming)面向切面编程,一种编程范式,指导开发者如何组织程序结构。
1.AOP的作用:
在不惊动原始设计的基础上为其进行功能增强(代理模式也可以实现这样的功能)。
2.Spring理念:
无入侵式/无侵入式
3.AOP核心概念:
- 连接点(JoinPoint):程序执行过程中的任意位置,粒度为执行方法、抛出异常、设置变量等
- 在SpringAOP中,理解为方法的执行
- 切入点(Pointcut):匹配连接点的式子
- 在SpringAOP中,一个切入点可以描述一个具体方法,也可也匹配多个方法
- 一个具体的方法:如com.dao包下的BookDao接口中的无形参无返回值的save方法
- 匹配多个方法:所有的save方法,所有的get开头的方法,所有以Dao结尾的接口中的任意方法,所有带有一个参数的方法
- 连接点范围要比切入点范围大,是切入点的方法也一定是连接点,但是是连接点的方法就不一定要被增强,所以可能不是切入点。
- 在SpringAOP中,一个切入点可以描述一个具体方法,也可也匹配多个方法
- 通知(Advice):在切入点处执行的操作,也就是共性功能
- 在SpringAOP中,功能最终以方法的形式呈现
- 通知类:定义通知的类
- 切面(Aspect):描述通知与切入点的对应关系。
- 目标对象(Target):原始功能去掉共性功能对应的类产生的对象,这种对象是无法直接完成最终工作的,目标对象就是要增强的类,也叫原始对象。
- 代理(Proxy):目标对象无法直接完成工作,需要对其进行功能回填,通过原始对象的代理对象实现。
4.AOP的切入点表达式
(1).语法格式
对于切入点表达式的语法为:
- 切入点表达式标准格式:动作关键字(访问修饰符 返回值 包名.类/接口名.方法名(参数) 异常名)
对于这个格式,通过一个例子,理解它:
execution(public User com.itzjy.service.UserService.findById(int))
- execution:动作关键字,描述切入点的行为动作,例如execution表示执行到指定切入点
- public:访问修饰符,还可以是public,private等,可以省略
- User:返回值,写返回值类型
- com.itheima.service:包名,多级包使用点连接
- UserService:类/接口名称
- findById:方法名
- int:参数,直接写参数的类型,多个类型用逗号隔开
- 异常名:方法定义中抛出指定异常,可以省略
(2).通配符
-
*:单个独立的任意符号,可以独立出现,也可以作为前缀或者后缀的匹配符出现
execution(public * com.itzjy.*.UserService.getBy*(*))
匹配com.itzjy包下的任意包中的UserService类或接口中所有fgetBy开头的带有一个参数的方法
-
…:多个连续的任意符号,可以独立出现,常用于简化包名与参数的书写
execution(public User com..UserService.findById(..))
匹配com包下的任意包中的UserService类或接口中所有名称为findById的方法
-
+
:专用于匹配子类类型execution(* *..*Service+.*(..))
这个使用率较低,描述子类的。*Service+,表示所有以Service结尾的接口的子类。
举例子:
execution(void com.itzjy.dao.BookDao.update())
匹配接口
execution(void com.itzjy.dao.impl.BookDaoImpl.update())
匹配实现类
execution(* com.itzjy.dao.impl.BookDaoImpl.update())
返回值任意
execution(* com.itzjy.dao.impl.BookDaoImpl.update(*))
返回值任意,但是update方法必须要有一个参数
execution(void com.*.*.*.*.update())
返回值为void,com包下的任意包三层包下的任意类的update方法
execution(void com.*.*.*.update())
返回值为void,com包下的任意两层包下的任意类的update方法
execution(void *..update())
返回值为void,方法名是update的任意包下的任意类
execution(* *..*(..))
匹配项目中任意类的任意方法,能匹配,但是不建议使用这种方式,影响范围广
execution(* *..u*(..))
匹配项目中任意包任意类下只要以u开头的方法
execution(* *..*e(..))
匹配项目中任意包任意类下只要以e结尾的方法
execution(void com..*())
返回值为void,com包下的任意包任意类任意方法,*代表的是方法
execution(* com.itzjy.*.*Service.find*(..))
将项目中所有业务层方法的以find开头的方法匹配
5.AOP通知类型
AOP共提供了5种通知类型:
- **前置通知(@Before):**追加功能到方法执行前,类似于在代码1或者代码2添加内容
- **后置通知(@After):**追加功能到方法执行后,不管方法执行的过程中有没有抛出异常都会执行,类似于在代码5添加内容
- **环绕通知(@Around):**环绕通知功能比较强大,它可以追加功能到方法执行的前后,这也是比较常用的方式,它可以实现其他四种通知类型的功能
- **返回后通知(@AfterReturning):**追加功能到方法执行后,只有方法正常执行结束后才进行,类似于在代码3添加内容,如果方法执行抛出异常,返回后通知将不会被添加
- **抛出异常后通知(@AfterThrowing):**追加功能到方法抛出异常后,只有方法执行出异常才进行,类似于在代码4添加内容,只有方法抛出异常后才会被添加
6.AOP通知中获取参数
- 获取切入点方法的参数,所有的通知类型都可以获取参数
- JoinPoint:适用于前置、后置、返回后、抛出异常后通知
- ProceedingJoinPoint:适用于环绕通知
- 获取切入点方法返回值,前置和抛出异常后通知是没有返回值,后置通知可有可无
- 返回后通知
- 环绕通知(Object)
- 获取切入点方法运行异常信息,前置和返回后通知是不会有,后置通知可有可无
- 抛出异常后通知
- 环绕通知