SpringAop @Aspect织入不生效,不执行前置增强织入@Before方式

作者:苦作舟 时间:2021-09-18 04:00:58 

SpringAop @Aspect织入不生效,不执行前置增强织入@Before

想写一个AOP,主要有2个用意

  • 第一个用意是做后端的防表单重复提交的token验证。

  • 第二个用意是对后台JSR303 Validator的校验结果做一个统一处理,不想把对校验结果的处理分散在每个controller方法中


@ResponseBody
@RequestMapping(value = "add", method = RequestMethod.POST)
public ResponseModel add(@Valid User user, BindingResult br, HttpServletResponse response) {

if(br.hasErrors()) {
return ResponseModel.validFail(getErrorsSplitNewLine(br));
}
accountService.addUser(user);
return ResponseModel.success("保存用户成功");
}

如上面方法中, br.hasErrors() 在每个表单提交方法中都存在,想单独抽出来使用AOP统一处理。

所以写一个AOP,如下:


@Aspect
@Component
public class ParamValidAspect {
   @Before("@annotation(com.hebao.tech.adm.framework.annotation.ParamValid)")
   public void paramValid(JoinPoint point) {
System.out.println("参数校验切入方法被调用了.....");
       //省略
   }
}

由于这篇文章主要是记录AOP不生效的原因,所以,这里不写具体实现了。

上面的内容定义一个Aop织入,在有注解@ParamValid的注释Controller方法上织入。


import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ParamValid {

}

这个ParamValid的内容,仅仅是一个标志性的注解,声明为方法层的注解,并且是运行时注解。

最后在application.xml中加入AOP * 设置。


<!-- 这个配置要配置在component-scan以后 -->
<aop:aspectj-autoproxy proxy-target-class="true" />

如果spring配置文件没引入过aop的配置,还需要在加入xml声明

SpringAop @Aspect织入不生效,不执行前置增强织入@Before方式

大功告成,测试了一下,发现有点悲剧,根本织入不生效,也不报错,,楞是不执行相关的织入代码。

最后在网上搜了一下,发现Spring与SpringMVC是2个不同的父子容器, @Aspect如果被spring容器加载的话,而@Controller注解的这些类的实例化以及注入却是由SpringMVC来完成。 @Aspect如果被spring容器加载的时候,可能Spring MVC容器还未初始化, Controller类还未初始化,所以无法正常织入。。

所以调整如下:


@Aspect
public class ParamValidAspect {
   @Before("@annotation(com.hebao.tech.adm.framework.annotation.ParamValid)")
   public void paramValid(JoinPoint point) {
System.out.println("参数校验切入方法被调用了.....");
       //省略
   }
}

去掉@Component注解,然后把 aop:aspectj-autoproxy 移入springmvc配置文件中,并定义bean,如下:


<!-- 这个配置一定要配置在component-scan以后 -->
<aop:aspectj-autoproxy proxy-target-class="true" />
<bean id="paramValidAspect" class="com.hebao.tech.adm.framework.spring.aop.ParamValidAspect"/>

这样就大功告成了。

使用@Aspect,@Before不被调用


@Aspect
@Component
public class LogAspect {
   @Before("pointcut()")
   public void before(){
       System.out.println("before");
   }

@Pointcut("@annotation(com.demo.annotation.Log)")
   public void pointcut(){    
   }

@Around("pointcut()")
   public void around(){
       System.out.println("arount");
   }

@After("pointcut()")
   public void after(){
       System.out.println("after");
   }
}

调用方法返回结果:

arount

after


@Aspect
@Component
public class LogAspect {
   @Before("pointcut()")
   public void before(){
       System.out.println("before");
   }

@Pointcut("@annotation(com.mxy.annotation.Log)")
   public void pointcut(){

}

@Around("pointcut()")
   public void around(ProceedingJoinPoint point){
       System.out.println("arount before");

try {
           point.proceed();
       } catch (Throwable throwable) {
           throwable.printStackTrace();
       }
       System.out.println("arount after");
   }

@After("pointcut()")
   public void after(){
       System.out.println("after");
   }
}

调用返回结果:

arount before

before

arount after

after

来源:https://blog.csdn.net/oKuZuoZhou/article/details/81015310

标签:SpringAop,@Aspect织入,织入@Before
0
投稿

猜你喜欢

  • Java实现Http工具类的封装操作示例

    2021-08-14 10:27:57
  • Java设计模式之Strategy模式

    2023-11-21 03:58:22
  • Android使用AIDL方式实现播放音乐案例

    2022-01-04 23:58:40
  • springboot框架阿里开源低代码工具LowCodeEngine

    2022-09-01 09:40:41
  • android手机获取唯一标识的方法

    2022-05-28 19:24:38
  • C#递归算法之归并排序

    2023-01-01 14:49:36
  • Android仿美团淘宝实现多级下拉列表菜单功能

    2022-07-24 18:42:18
  • SpringBoot下的值注入(推荐)

    2023-04-01 04:45:01
  • spring-boot-autoconfigure模块用法详解

    2023-11-25 12:59:19
  • java中thread线程start和run的区别

    2023-09-30 13:12:16
  • JSON序列化Redis读取出错问题解决方案

    2022-10-13 18:57:50
  • C#读取静态类常量属性和值的实例讲解

    2022-04-15 16:37:34
  • Java中new关键字和newInstance方法的区别分享

    2022-12-04 03:12:52
  • Android学习笔记-保存数据到SQL数据库中(Saving Data in SQL Databases)

    2023-07-28 08:32:20
  • Java使用openssl检测网站是否支持ocsp

    2022-10-03 15:55:55
  • Java C++题解leetcode字符串轮转KMP算法详解

    2023-05-30 11:28:49
  • C# JWT权限验证的实现

    2022-11-24 00:57:13
  • Gradle配置教程之自定义APK名称与输出路径

    2023-03-04 17:35:27
  • Android Studio配置国内镜像源(利用hosts)

    2023-05-29 18:47:54
  • Android多线程学习实例详解

    2022-02-17 19:00:56
  • asp之家 软件编程 m.aspxhome.com