首頁 > 軟體

Spring this呼叫當前類方法無法攔截的範例程式碼

2022-03-20 19:00:44

先給出程式碼範例

package com.example.demo.service;
import org.springframework.stereotype.Service;
@Service
public class ProxyService {
    public void  testA(){
        System.out.println("進入A");
        this.testB();
    }
    public void testB() {
        System.out.println("進入b");
    }

}
package com.example.demo.annotation;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
@Aspect
@Component
public class AspectjTest {
    @Around("execution(* com.example.demo.service.ProxyService.testB())")
    public void recordProxy(ProceedingJoinPoint joinPoint) throws Throwable {
        long start = System.currentTimeMillis();
        joinPoint.proceed();
        long end = System.currentTimeMillis();
        System.out.println("花費時間:"+(end-start));
    }
}
package com.example.demo.api;
import com.example.demo.service.ProxyService;
import com.example.demo.service.UserService;
import org.springframework.aop.framework.AopContext;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.PostMapping;
@Controller
public class ProxyApi {
//    @Autowired
//    ProxyService proxyService1;
    @Autowired
    private ApplicationContext applicationContext;
    @PostMapping("/proxy")
    public String test1() {
        ProxyService proxyService1 =  applicationContext.getBean(ProxyService.class);;
        proxyService1.testA();
        return "string";
    }
}

執行上面的程式碼會發現 設定aop 攔截方法不會被執行

我們通過debug 檢視這個proxyService1 和this的區別,看看他們的值是什麼

發現不一樣,其實這就是問題的原因。

1、當我們在aop設定攔截的時候會指定類下面的方法路徑,在spring啟動的時候會先去載入這個ProxyService類,生成一個bean,但是因為你用aop設定了,所以需要代理這個ProxyService類,所以最終存在spring容器中的bean物件就是被代理後的bean物件。所以,我們在用容器獲取bean或者用依賴注入獲取bean的地址路徑顯示的是被代理後的bean 。
2、this獲取的當前物件方法的一個參照,所以在呼叫testB方法的時候用的不是被代理的物件,自熱不會經過aop攔截,原理和我們使用普通動態代理一樣,只能是代理物件才能走自定義的方法。
3、可以通過debug 檢視當ProxyService類被代理前和後的zhi值

發現是和之前的debug截圖裡面的值相符合的哈。

解決方法,就是在呼叫testB方法的時候用spring容器裡的bean物件

@Service
public class ProxyService {
    @Autowired
    private  ProxyService proxyService;
    
    public void  testA(){
        System.out.println("進入A");
        proxyService.testB();
    }
    public void testB() {
        System.out.println("進入b");
}

或者

@Service
public class ProxyService {
    public void  testA(){
        System.out.println("進入A");
        ProxyService currentProxy = (ProxyService) AopContext.currentProxy();
        currentProxy.testB();
    }
    public void testB() {
        System.out.println("進入b");
    }
}

最終結果正確

到此這篇關於Spring this呼叫當前類方法無法攔截的文章就介紹到這了,更多相關Spring this無法攔截內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com