詳解spring中aop不生效的幾種解決辦法
先看下這個(gè)問(wèn)題的背景:假設(shè)有一個(gè)spring應(yīng)用,開(kāi)發(fā)人員希望自定義一個(gè)注解@Log,可以加到指定的方法上,實(shí)現(xiàn)自動(dòng)記錄日志(入?yún)ⅰ⒊鰠?、響?yīng)耗時(shí)這些)
package com.cnblogs.yjmyzz.springbootdemo.aspect;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Log {
}
然后再寫一個(gè)Aspect來(lái)解析這個(gè)注解,對(duì)打了Log注解的方法進(jìn)行增強(qiáng)處理
package com.cnblogs.yjmyzz.springbootdemo.aspect;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
import java.lang.reflect.Method;
@Component
@Aspect
public class LogAspect {
@Pointcut("execution (* com.cnblogs.yjmyzz.springbootdemo.service..*.*(..))")
public void logPointcut() {
}
@Around("logPointcut()")
public void around(JoinPoint point) {
String methodName = point.getSignature().getName();
Object[] args = point.getArgs();
Class<?>[] argTypes = new Class[point.getArgs().length];
for (int i = 0; i < args.length; i++) {
argTypes[i] = args[i].getClass();
}
Method method = null;
try {
method = point.getTarget().getClass().getMethod(methodName, argTypes);
} catch (Exception e) {
e.printStackTrace();
}
//獲取方法上的注解
Log log = method.getAnnotation(Log.class);
if (log != null) {
//演示方法執(zhí)行前,記錄一行日志
System.out.println("before:" + methodName);
}
try {
//執(zhí)行方法
((ProceedingJoinPoint) point).proceed();
} catch (Throwable throwable) {
throwable.printStackTrace();
} finally {
if (log != null) {
//演示方法執(zhí)行后,記錄一行日志
System.out.println("after:" + methodName);
}
}
}
}
寫一個(gè)測(cè)試Service類:
package com.cnblogs.yjmyzz.springbootdemo.service;
import com.cnblogs.yjmyzz.springbootdemo.aspect.Log;
import org.springframework.stereotype.Component;
@Component
public class HelloService {
@Log
public void sayHi(String msg) {
System.out.println("\tsayHi:" + msg);
}
public void anotherSayHi(String msg) {
this.sayHi(msg);
}
}
最后來(lái)跑一把:
package com.cnblogs.yjmyzz.springbootdemo;
import com.cnblogs.yjmyzz.springbootdemo.service.HelloService;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
/**
* @author 菩提樹(shù)下的楊過(guò)
*/
@ComponentScan("com.cnblogs.yjmyzz")
@Configuration
@EnableAspectJAutoProxy
public class SampleApplication {
public static void main(String[] args) {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(SampleApplication.class);
HelloService helloService = context.getBean(HelloService.class);
helloService.sayHi("hi-1");
System.out.println("\n");
helloService.anotherSayHi("hi-2");
}
}
輸出如下:

顯然HelloService中的anotherSayHi方法,并未被aop增強(qiáng)。 原因其實(shí)很簡(jiǎn)單,了解AOP原理的同學(xué)想必都知道,AOP的實(shí)現(xiàn)有二類,如果是基于接口的,會(huì)采用動(dòng)態(tài)代理,生成一個(gè)代理類,如果是基于類的,會(huì)采用CGLib生成子類,然后在子類中擴(kuò)展父類中的方法。

本文中HelloService并不是一個(gè)接口,所以從上圖的斷點(diǎn)中可以看出,當(dāng)Spring運(yùn)行時(shí),HelloService被增加為...EnhancerBySpringCGLib...。但是當(dāng)調(diào)用到anotherSayHi時(shí)

方法的調(diào)用方,其實(shí)是原始的HelloSerfvice實(shí)例,即:是未經(jīng)過(guò)Spring AOP增強(qiáng)的對(duì)象實(shí)例。所以解決問(wèn)題的思路就有了,想辦法用增強(qiáng)后的HelloService實(shí)例來(lái)調(diào)用!
方法一:用Autowired 注入自身的實(shí)例

這個(gè)方法,第一眼看上去感覺(jué)有些怪,自己注入自己,感覺(jué)有點(diǎn)象遞歸/死循環(huán)的搞法,但確實(shí)可以work,Spring在解決循環(huán)依賴上有自己的處理方式,避免了死循環(huán)。
方法二:從Spring上下文獲取增強(qiáng)后的實(shí)例引用

原理與方法一其實(shí)類似,不多解釋。
方法三: 利用AopContext

不過(guò)這個(gè)方法要注意的是,主類入口上,必須加上exporseProxy=true,參考下圖:

最后來(lái)驗(yàn)證下這3種方法是否生效:

從運(yùn)行結(jié)果上看,3種方法都可以解決這個(gè)問(wèn)題?!?/p>
到此這篇關(guān)于詳解spring中aop不生效的幾種解決辦法的文章就介紹到這了,更多相關(guān)spring中aop不生效內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
作者:菩提樹(shù)下的楊過(guò)
出處:http://yjmyzz.cnblogs.com
相關(guān)文章
Mybatis框架之代理模式(Proxy Pattern)的實(shí)現(xiàn)
本文主要介紹了MyBatis框架中使用代理模式ProxyPattern的原理和實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-11-11
SpringBoot實(shí)現(xiàn)圖片上傳及本地訪問(wèn)
在SpringBoot項(xiàng)目中,處理靜態(tài)文件訪問(wèn)尤其是實(shí)時(shí)更新的文件如商品圖片,可通過(guò)配置WebMvcConfig將本地文件映射到URL路徑上,以解決重啟項(xiàng)目才能訪問(wèn)文件的問(wèn)題,本文詳解如何保存和訪問(wèn)這些文件,幫助開(kāi)發(fā)者優(yōu)化項(xiàng)目文件管理2022-09-09
Java正則驗(yàn)證字串符RegexValidator類使用
正則驗(yàn)證字串符是一種強(qiáng)大的工具,可以幫助程序員在處理字符串時(shí)輕松進(jìn)行復(fù)雜匹配,本文將介紹正則表達(dá)式的概念、語(yǔ)法和在編程中的應(yīng)用,并通過(guò)實(shí)例演示如何使用正則表達(dá)式進(jìn)行字符串匹配、替換和提取等操作2023-11-11
深入解析Spring?TransactionTemplate?高級(jí)用法(示例代碼)
TransactionTemplate是?Spring?框架中一個(gè)強(qiáng)大的工具,它允許開(kāi)發(fā)者以編程方式控制事務(wù),通過(guò)本文的詳細(xì)解析和示例代碼,我們可以看到如何使用?`TransactionTemplate`?來(lái)執(zhí)行事務(wù)性操作,并處理高級(jí)事務(wù)場(chǎng)景,感興趣的朋友跟隨小編一起看看吧2025-02-02
Spring Boot整合MyBatis-Plus實(shí)現(xiàn)CRUD操作的示例代碼
本文主要介紹了Spring Boot整合MyBatis-Plus實(shí)現(xiàn)CRUD操作,可以快速實(shí)現(xiàn)數(shù)據(jù)庫(kù)的增刪改查操作,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2025-04-04
SpringBoot項(xiàng)目如何訪問(wèn)jsp頁(yè)面的示例代碼
本篇文章主要介紹了SpringBoot項(xiàng)目如何訪問(wèn)jsp頁(yè)面的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-10-10
java 中的static關(guān)鍵字和final關(guān)鍵字的不同之處
java 中的static關(guān)鍵字和final關(guān)鍵字的不同之處,需要的朋友可以參考一下2013-03-03
springboot vue組件開(kāi)發(fā)實(shí)現(xiàn)接口斷言功能
這篇文章主要為大家介紹了springboot+vue組件開(kāi)發(fā)實(shí)現(xiàn)接口斷言功能,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05
Java實(shí)現(xiàn)滑動(dòng)窗口算法的示例代碼
滑動(dòng)窗口算法是一種高效解決子數(shù)組、子字符串問(wèn)題的算法,廣泛應(yīng)用于數(shù)據(jù)流處理、網(wǎng)絡(luò)限流和字符串操作等場(chǎng)景,本文將詳細(xì)解析滑動(dòng)窗口算法的核心思想、常見(jiàn)問(wèn)題及其實(shí)現(xiàn)方式,需要的朋友可以參考下2025-03-03

