溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

SpringAop中如何實(shí)現(xiàn)@Aspect織入不生效、不執(zhí)行前置增強(qiáng)織入@Before

發(fā)布時(shí)間:2021-12-02 11:43:47 來(lái)源:億速云 閱讀:205 作者:小新 欄目:開(kāi)發(fā)技術(shù)

這篇文章給大家分享的是有關(guān)SpringAop中如何實(shí)現(xiàn)@Aspect織入不生效、不執(zhí)行前置增強(qiáng)織入@Before的內(nèi)容。小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過(guò)來(lái)看看吧。

    SpringAop @Aspect織入不生效,不執(zhí)行前置增強(qiáng)織入@Before

    想寫(xiě)一個(gè)AOP,主要有2個(gè)用意

    • 第一個(gè)用意是做后端的防表單重復(fù)提交的token驗(yàn)證。

    • 第二個(gè)用意是對(duì)后臺(tái)JSR303 Validator的校驗(yàn)結(jié)果做一個(gè)統(tǒng)一處理,不想把對(duì)校驗(yàn)結(jié)果的處理分散在每個(gè)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("保存用戶(hù)成功");
    	}

    如上面方法中, br.hasErrors() 在每個(gè)表單提交方法中都存在,想單獨(dú)抽出來(lái)使用AOP統(tǒng)一處理。

    所以寫(xiě)一個(gè)AOP,如下:

    @Aspect
    @Component
    public class ParamValidAspect {
        @Before("@annotation(com.hebao.tech.adm.framework.annotation.ParamValid)")
        public void paramValid(JoinPoint point) {
    	System.out.println("參數(shù)校驗(yàn)切入方法被調(diào)用了.....");
            //省略
        } 
    }

    由于這篇文章主要是記錄AOP不生效的原因,所以,這里不寫(xiě)具體實(shí)現(xiàn)了。

    上面的內(nèi)容定義一個(gè)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 {
     
    }

    這個(gè)ParamValid的內(nèi)容,僅僅是一個(gè)標(biāo)志性的注解,聲明為方法層的注解,并且是運(yùn)行時(shí)注解。

    最后在application.xml中加入AOP動(dòng)態(tài)代理設(shè)置。

    <!-- 這個(gè)配置要配置在component-scan以后 -->
    <aop:aspectj-autoproxy proxy-target-class="true" />

    如果spring配置文件沒(méi)引入過(guò)aop的配置,還需要在加入xml聲明

    SpringAop中如何實(shí)現(xiàn)@Aspect織入不生效、不執(zhí)行前置增強(qiáng)織入@Before

    大功告成,測(cè)試了一下,發(fā)現(xiàn)有點(diǎn)悲劇,根本織入不生效,也不報(bào)錯(cuò),,楞是不執(zhí)行相關(guān)的織入代碼。

    最后在網(wǎng)上搜了一下,發(fā)現(xiàn)Spring與SpringMVC是2個(gè)不同的父子容器, @Aspect如果被spring容器加載的話,而@Controller注解的這些類(lèi)的實(shí)例化以及注入?yún)s是由SpringMVC來(lái)完成。 @Aspect如果被spring容器加載的時(shí)候,可能Spring MVC容器還未初始化, Controller類(lèi)還未初始化,所以無(wú)法正??椚?。。

    所以調(diào)整如下:

    @Aspect
    public class ParamValidAspect {
        @Before("@annotation(com.hebao.tech.adm.framework.annotation.ParamValid)")
        public void paramValid(JoinPoint point) {
     System.out.println("參數(shù)校驗(yàn)切入方法被調(diào)用了.....");
            //省略
        } 
    }

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

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

    這樣就大功告成了。

    使用@Aspect,@Before不被調(diào)用

    @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"); 
        }
    }

    調(diào)用方法返回結(jié)果:

    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"); 
        }
    }

    調(diào)用返回結(jié)果:

    arount before

    before

    arount after

    after

    感謝各位的閱讀!關(guān)于“SpringAop中如何實(shí)現(xiàn)@Aspect織入不生效、不執(zhí)行前置增強(qiáng)織入@Before”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí),如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!

    向AI問(wèn)一下細(xì)節(jié)

    免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

    AI