AOP详解与编码实例

文章介绍了如何使用SpringAOP进行日志记录,通过构造切面在Controller方法上设置切点,记录请求信息。同时,展示了如何处理部门参数,通过定义切面和环绕通知减少代码改动。文章详细阐述了AOP的五大通知类型、执行顺序以及织入过程,并给出了具体的Aspect实现示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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..*.*(..))")

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值