亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Spring aop失效的幾種解決方案

瀏覽:80日期:2023-09-06 13:16:42

先看下這個(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 { }

然后再寫(xiě)一個(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@Aspectpublic 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); } } }}

寫(xiě)一個(gè)測(cè)試Service類(lèi):

package com.cnblogs.yjmyzz.springbootdemo.service; import com.cnblogs.yjmyzz.springbootdemo.aspect.Log;import org.springframework.stereotype.Component; @Componentpublic 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@EnableAspectJAutoProxypublic 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'); }}

輸出如下:

Spring aop失效的幾種解決方案

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

Spring aop失效的幾種解決方案

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

Spring aop失效的幾種解決方案

方法的調(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í)例

Spring aop失效的幾種解決方案

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

方法二:從Spring上下文獲取增強(qiáng)后的實(shí)例引用

Spring aop失效的幾種解決方案

原理與方法一其實(shí)類(lèi)似,不多解釋。

方法三: 利用AopContext

Spring aop失效的幾種解決方案

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

Spring aop失效的幾種解決方案

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

Spring aop失效的幾種解決方案

從運(yùn)行結(jié)果上看,3種方法都可以解決這個(gè)問(wèn)題。 

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。

標(biāo)簽: Spring
相關(guān)文章:
主站蜘蛛池模板: 精品国产看高清国产毛片 | 国产精品福利在线观看秒播 | 2014av手机天堂网 | 国产精品观看在线亚洲人成网 | 欧美日韩亚洲精品一区二区 | 国产免费大片 | 亚洲高清综合 | 国产成人夜色影视视频 | 免费大黄网站在线观看 | 久久国产精品高清一区二区三区 | 第四色婷婷基地 | 亚洲欧美国产日产综合不卡 | 九九热精品视频在线播放 | 多女多p多杂交视频在线观看 | 国产老妇一性一交一乱 | 亚洲国产天堂久久九九九 | 色婷婷五 | 日本大片免a费观看视频+播放器 | 免费观看性行为的视频网站 | 女人牲交视频一级毛片 | 美女18一级毛片免费看 | 国产综合视频在线观看 | 免费特黄一区二区三区视频一 | 欧美国产日本 | 成人亚洲精品一区二区 | 久久久久久久久女黄9999 | 片在线观看视频免费 | 国产入口在线观看 | 国产又黄又爽又色视频观看免费 | 亚洲色图视频在线 | 九九99久久精品在免费线bt | 国产性夜夜性夜夜爽91 | 99精品久久秒播无毒不卡 | 性亚洲精品 | 奇米影视亚洲狠狠色 | 全免费a级毛片免费看 | 亚洲精品一区二区深夜福利 | 久久久鲁| 日本人xxxxxxx中国 | 国产在线精品一区二区夜色 | 国产亚洲精品久久久999小说 |