早睡身体好!SpringAOP学习笔记

记录这篇文章是因为在项目开发过程中,需要建立一个日志模块。设想如果每个功能都需要添加一个添加日志的逻辑,就算能运用面向对象编程思想,将方法封装,是大大的增加了代码复用率。但是也需要每个方法添加封装的方法,一样是不能接受的。后续经过查资料,知道了AOP的用法,果然AOP就是为了日志管理、权限管理这些公用的功能而生的。

什么是SpringAOP?

AOP(Aspect Oriented Programing)面向切面编程。在整个程序贯穿之下,存在N个切点,而切点会连接着横切面。若程序执行到了连接点后,就会进入到横切面中执行你的逻辑增强。整个程序中,面向对象编程是纵向编程(接口、实现类)等而AOP是在原先程序的基础上,增加了非业务功能,而不是在业务逻辑中再添加各种非业务逻辑。

AOP相关注解

@Pointcut

该注解是指定一个切点,拦截你需要拦截的东西,在这个切点处进入切面,两种常用的表达式:execution()和annotation()

execution(* * com.service..*(..))

该表达式的意思是,监听了com.service包和其下的子包里的所有方法,包含所有参数,其中两个句点..指的是service包下的子包。

第一个*号表示返回类型,*表示所有返回类型

第二个*号表示类名,*表示所有类名

第三个*号表示方法,而(..)两个句点表示了所有参数

@annotation(com.annotation.MyAnnotation)

该表达式的意思是,监听所有加上了MyAnnotation注解的方法,MyAnnotation可以使自定义注解也可以是RequestMapping、GetMapping等注解。

若使用execution表达式觉得范围太大,则可以使用annotation表达式,就是需要自己去配置注解,略显麻烦。

可以在@Before、@After等注解中直接使用,也可以编写一个方法,使用 @Pointcut注解。

   @Pointcut("@annotation(com.headyonder.dmp.aop.annotation.SystemLogAnno)")
    public void activePointCut() {
    }

@Before

前置播报

就是监听切点定义的所有方法,在监听的方法进入切面之前就执行你的逻辑操作。

@Before(value = "activePointCut()")
public void doAfterReturning(JoinPoint jp){
            User u = UserUtil.getUser();
            MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
            String method = methodSignature.getName();
            SystemLogAnno anno = methodSignature.getMethod().getAnnotation(SystemLogAnno.class);
            log.info("=={}已经操作{}方法==", u.getName(), method);
}

@After

后置绝对播报

就是监听切点定义的所有方法,在监听的方法完成后(无论是否出现异常)都会进入切面执行你的逻辑操作。

@After(value = "activePointCut()")
public void doAfterReturning(JoinPoint jp){
            User u = UserUtil.getUser();
            MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
            String method = methodSignature.getName();
            SystemLogAnno anno = methodSignature.getMethod().getAnnotation(SystemLogAnno.class);
            log.info("=={}已经操作{}方法==", u.getName(), method);
}

@AfterReturing

后置完成播报

和After不一样的是,AfterReturing只会在方法成功执行并返回参数的时候才会执行你的逻辑增强。

@AfterReturning(value = "activePointCut()", returning = "result")
public void doAfterReturning(JoinPoint jp,String result){
            User u = UserUtil.getUser();
            MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
            String method = methodSignature.getName();
            SystemLogAnno anno = methodSignature.getMethod().getAnnotation(SystemLogAnno.class);
            log.info("=={}已经操作{}方法==", u.getName(), method);
}

需要注意的是,注解中returning的值必须与方法的第二个参数一致,否则程序会编译错误。 

@AfterThrowing

异常播报

监听方法进入切面后,抛出的异常执行你的逻辑增强。

@AfterThrowing(value = "activePointCut()", throwing = "e")
public void doThrowing(JoinPoint joinPoint, Exception e) {
       //与doAfterReturing类似
}

@Around

环绕播报(全程播报)

该注解的优先级大于Before和After。该注解的方法第一个参数需要为ProceedingJoinPoint,而其中最重要的是proceed()方法,该方法是执行目标对象。

就可以在该注解下进行前置增强和后置增强。

  @Around(value = "activePointCut()&&args(argNames,argName1)",argNames = "pjp,argNames,argName1")
    public void doAround(ProceedingJoinPoint pjp,String argNames,String argName1){
        try{
            MethodSignature methodSignature = (MethodSignature) pjp.getSignature();
            String method = methodSignature.getName();
            SystemLogAnno anno = methodSignature.getMethod().getAnnotation(SystemLogAnno.class);
            log.info("执行目标方法前");
            pjp.proceed(pjp.getArgs());
            log.info("执行目标方法后");
            //pjp.proceed();
        }catch (Throwable throwable) {
            throwable.printStackTrace();
        }
    }

注:每个逻辑增强中,都可以增加参数。但是需要在value中使用&&args(args1,args2,args3...)然后第二个参数,argNames就需要添加方法中所有的参数。如下:

@Around(value = "activePointCut()&&args(argNames,argName1)",argNames = "pjp,argNames,argName1")
 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值