1. AOP
面向切面编程
小白可以理解为,其实就是一个函数里面执行了另一个函数,
顺便写一些其他逻辑代码
2. aop概念及优点
AOP场景:
记录操作日志,权限控制,事务管理
AOP优势:
代码无入侵,减少重复代码,提高并发效率,维护方便
3. 步骤:
引入依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
1. 定义AOP类,加注解@component和@accept
表示当前类交给IOC容器管理的bean对象,并且是AOP类
2. 然后在AOP类里定义方法,实现要做的事情
3. 然后在改方法上加上AOP通知注解
4. AOP通知注解
1. @Around: 环绕通知,目标方法前后都会执行
2. @Before 前置通知
3. @After 目标方法执行后执行,有异常不会执行(重点)
4. @AfterReturning 返回后通知,有异常不执行(了解)
5. @AfterThrowing 异常后通知(了解)
注意事项:
@Around 环绕通知需要自己调用 ProceedingJoinPoint.proceed()
来让原始方法执行,其他通知不需要考虑目标方法执行
@Around 返回值必须是Object,来接收原始方法的返回值
5. AOP通知顺序
默认按类名字母排序
需要用注解 @Order(数字) 控制顺序
6、AOP切入点表达式
切入点表达式,就是为了,控制AOP类,在那些地方生效
- execution
execution(访问修饰符? 返回值 包名.类名?.方法名(方法参数) throws异常?)
访问修饰符 一般都省略
包名 类名 可省略,但是一般不建议省略
throws异常 一般都省略
可以使用通配符描述切入点
* 表示所有 返回值,包名,类名,方法名,任意类型一个参数
.. 表示所有层级的包or任意个数的任意参数
@Pointcut("execution(* com.*.*.update(*))")
@Pointcut("execution(* com.example.boot3..update(..)))")
@Pointcut("execution(* com.sky.mapper.*.*(..)) &&
@annotation(com.sky.annotation.AutoFill)")
也可以用&& || ! 来组合比较复杂的表达式
书写建议:
所有业务方法名在命名时尽量规范,方便切入点表达式快速匹配
描述切入点方法通常基于接口描述,而不是直接描述实现类,增强拓展性
在满足业务需要的前提下,尽量缩小切入点的匹配范围
- @annotation
第一步:定义—自定义注解
基于注解的形式,来控制切入点
//右键 new class 创建一个Annotation,class名随便写
//描述这个注解什么时候生效的
//@Retention是用来修饰注解的,注解的注解,也称为元注解
//修饰这个注解生效的时机
@Retention(RetentionPolicy.RUNTIME)
//修饰这个注解生效的时机
@Target(ElementType.METHOD)
//@interface 表示当前类是一个注解类
public @interface MyLog {
}
第二步:
@Aspect
@Component
public class MyAspect {
@Pointcut("annotation(com.itheima.aop.MyLog)")
public void serviceMethods() {}
@Before("serviceMethods()")
public void beforeServiceMethod(JoinPoint joinPoint) {
// 在 serviceMethods 定义的切点匹配的方法执行前执行通知
}
@After("serviceMethods()")
public void afterServiceMethod(JoinPoint joinPoint) {
// 在 serviceMethods 定义的切点匹配的方法执行后执行通知
}
}
7. AOP连接点
spring中joinpoint抽象了连接点,
用它可以获得方法执行时的相关信息,如目标类名,方法名,方法参数。
@Around通知,
使用ProceedingJoinPoint获取
其他四种通知,
获取连接点信息只能使用JoinPoint,
她是ProceedingJoinPoint的父类型
8、代码示例
定义AOP类
package com.example.boot3.aop;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
//@Component把这个类交给IOC容器管理,注册为bean对象
@Component
//@Aspect表示该类为 AOP类
@Aspect
public class MyAspect {
@Pointcut("execution(* com.example.service.*.*(..))")
public void serviceMethods() {}
@Before("serviceMethods()")
public void beforeServiceMethod(JoinPoint joinPoint) {
// 在 serviceMethods 定义的切点匹配的方法执行前执行通知
}
@After("serviceMethods()")
public void afterServiceMethod(JoinPoint joinPoint) {
// 在 serviceMethods 定义的切点匹配的方法执行后执行通知
}
}