1、日志:构造切面,在所有controller方法上设置切点,记录所有请求的url、参数、时间、账号等信息。
2、部门参数代码提取:项目底层只提供了部门id、改造要求需要提供部门编码、名称、等级信息,改代码工作量大,定义切面,通过在方法上加注解触发环绕通知,拼接部门信息到参数里,在交给业务代码层处理,改动量小。
aop五大通知的执行顺序
环绕、前置、后置、返回、异常
在一个方法只被一个aspect类拦截时:
1.1 正常情况下:环绕通知–>前置通知–>代码–>后置通知–>afterReturning
1.2 异常情况下:环绕通知–>前置通知–>异常代码–>后置通知–>AfterThrowing
同一个方法被多个Aspect类拦截
用Order注解指定每个 aspect 的执行顺序,值越小的 aspect 越先执行
连接点(Joinpoint) 程序执行的某个特定位置,如某个方法调用前,调用后,方法抛出异常后,这些代码中的特定点称为连接点。简单来说,就是在哪加入你的逻辑增强
连接点表示具体要拦截的方法,上面切点是定义一个范围,而连接点是具体到某个方法
切点(PointCut) 每个程序的连接点有多个,如何定位到某个感兴趣的连接点,就需要通过切点来定位。比如,连接点--数据库的记录,切点--查询条件
切点用于来限定Spring-AOP启动的范围,通常我们采用表达式的方式来设置,所以关键词是范围
增强(Advice) 增强是织入到目标类连接点上的一段程序代码。在Spring中,像BeforeAdvice等还带有方位信息
通知是直译过来的结果,我个人感觉叫做“业务增强”更合适 对照代码就是拦截器定义的相关方法,通知分为如下几种:
前置通知(before):在执行业务代码前做些操作,比如获取连接对象
后置通知(after):在执行业务代码后做些操作,无论是否发生异常,它都会执行,比如关闭连接对象
异常通知(afterThrowing):在执行业务代码后出现异常,需要做的操作,比如回滚事务
返回通知(afterReturning):在执行业务代码后无异常,会执行的操作
环绕通知(around):这个目前跟我们谈论的事务没有对应的操作,所以暂时不谈
目标对象(Target) 需要被加强的业务对象(利用自定义注解、execution指定的类/方法)
织入(Weaving) 织入就是将增强添加到对目标类具体连接点上的过程。
织入是一个形象的说法,具体来说,就是生成代理对象并将切面内容融入到业务流程的过程。
代理类(Proxy) 一个类被AOP织入增强后,就产生了一个代理类。
切面(Aspect) 切面由切点和增强组成,它既包括了横切逻辑的定义,也包括了连接点的定义,SpringAOP就是将切面所定义的横切逻辑织入到切面所制定的连接点中。
原理:利用动态代理实现,
@Aspect
@Component
@Order(1) // aspect 执行顺序
public class SystemLogAspect {
/* ①利用注解的方式进行织入,指定注解 */
@Pointcut("@annotation(com.sparrow.aspect.SystemLog)")
/* ②扫描com.sparrow.controller包下所有方法织入 */
@Pointcut("execution(* com.sparrow.controller..*.*(..))")
public void logAopAspect() { }
// 环绕通知
@Around("logAopAspect()")
public Object doAround(ProceedingJoinPoint joinPoint) {}
// 前置通知
@Before("logAopAspect()")
public void doBefore(JoinPoint joinPoint) {}
// 后置通知
@After("logAopAspect()")
public void doAfter(JoinPoint joinPoint) {}
// 返回通知
@AfterReturning(value="logAopAspect()", returning="result")
public void doReturn(JoinPoint joinPoint, Object result) {}
// 异常通知
@AfterThrowing(value="logAopAspect()", throwing="ex")
public void doThrowe(JoinPoint joinPoint, Exception ex) {}
}
织入方式
构造自定义注解,通过在方法上加自定义注解,实现显式织入切点。
通过在切面类的切点上指定自定义注解,如下指定自定义注解SystemLog
@Pointcut("@annotation(com.sparrow.aspect.SystemLog)")
通过模式匹配,扫描指定的目录、文件、方法,批量织入切点。
如下方式是扫描com.sparrow.controller文件夹,在所有类里的所有方法上织入切点
@Pointcut("execution(* com.sparrow.controller..*.*(..))")