电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

詳解spring中aop不生效的幾種解決辦法

瀏覽:6日期:2023-08-31 15:54:31

先看下這個問題的背景:假設有一個spring應用,開發人員希望自定義一個注解@Log,可以加到指定的方法上,實現自動記錄日志(入參、出參、響應耗時這些)

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 { }

然后再寫一個Aspect來解析這個注解,對打了Log注解的方法進行增強處理 

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) { //演示方法執行前,記錄一行日志 System.out.println('before:' + methodName); } try { //執行方法 ((ProceedingJoinPoint) point).proceed(); } catch (Throwable throwable) { throwable.printStackTrace(); } finally { if (log != null) {//演示方法執行后,記錄一行日志System.out.println('after:' + methodName); } } }}

寫一個測試Service類:

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

最后來跑一把:

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 菩提樹下的楊過 */@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增強。 原因其實很簡單,了解AOP原理的同學想必都知道,AOP的實現有二類,如果是基于接口的,會采用動態代理,生成一個代理類,如果是基于類的,會采用CGLib生成子類,然后在子類中擴展父類中的方法。

詳解spring中aop不生效的幾種解決辦法

本文中HelloService并不是一個接口,所以從上圖的斷點中可以看出,當Spring運行時,HelloService被增加為...EnhancerBySpringCGLib...。但是當調用到anotherSayHi時

詳解spring中aop不生效的幾種解決辦法

方法的調用方,其實是原始的HelloSerfvice實例,即:是未經過Spring AOP增強的對象實例。所以解決問題的思路就有了,想辦法用增強后的HelloService實例來調用!

方法一:用Autowired 注入自身的實例

詳解spring中aop不生效的幾種解決辦法

這個方法,第一眼看上去感覺有些怪,自己注入自己,感覺有點象遞歸/死循環的搞法,但確實可以work,Spring在解決循環依賴上有自己的處理方式,避免了死循環。

方法二:從Spring上下文獲取增強后的實例引用

詳解spring中aop不生效的幾種解決辦法

原理與方法一其實類似,不多解釋。

方法三: 利用AopContext

詳解spring中aop不生效的幾種解決辦法

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

詳解spring中aop不生效的幾種解決辦法

最后來驗證下這3種方法是否生效:

詳解spring中aop不生效的幾種解決辦法

從運行結果上看,3種方法都可以解決這個問題。 

到此這篇關于詳解spring中aop不生效的幾種解決辦法的文章就介紹到這了,更多相關spring中aop不生效內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

作者:菩提樹下的楊過出處:http://yjmyzz.cnblogs.com

標簽: Spring
相關文章:
主站蜘蛛池模板: 学考网学历中心| 众能联合-提供高空车_升降机_吊车_挖机等一站工程设备租赁 | 精密冲床,高速冲床等冲压设备生产商-常州晋志德压力机厂 | 铝板冲孔网,不锈钢冲孔网,圆孔冲孔网板,鳄鱼嘴-鱼眼防滑板,盾构走道板-江拓数控冲孔网厂-河北江拓丝网有限公司 | 河南不锈钢水箱_地埋水箱_镀锌板水箱_消防水箱厂家-河南联固供水设备有限公司 | EPK超声波测厚仪,德国EPK测厚仪维修-上海树信仪器仪表有限公司 | 过滤器_自清洗过滤器_气体过滤器_苏州华凯过滤技术有限公司 | 臭氧发生器_臭氧消毒机 - 【同林品牌 实力厂家】 | 同学聚会纪念册制作_毕业相册制作-成都顺时针宣传画册设计公司 | 哈尔滨京科脑康神经内科医院-哈尔滨治疗头痛医院-哈尔滨治疗癫痫康复医院 | 东莞动力锂电池保护板_BMS智能软件保护板_锂电池主动均衡保护板-东莞市倡芯电子科技有限公司 | 螺旋绞龙叶片,螺旋输送机厂家,山东螺旋输送机-淄博长江机械制造有限公司 | 土壤肥料养分速测仪_测土配方施肥仪_土壤养分检测仪-杭州鸣辉科技有限公司 | 老房子翻新装修,旧房墙面翻新,房屋防水补漏,厨房卫生间改造,室内装潢装修公司 - 一修房屋快修官网 | 智能终端_RTU_dcm_北斗星空自动化科技 | 振动筛,震动筛,圆形振动筛,振动筛价格,振动筛厂家-新乡巨宝机电 蒸汽热收缩机_蒸汽发生器_塑封机_包膜机_封切收缩机_热收缩包装机_真空机_全自动打包机_捆扎机_封箱机-东莞市中堡智能科技有限公司 | 数码听觉统合训练系统-儿童感觉-早期言语评估与训练系统-北京鑫泰盛世科技发展有限公司 | 中开泵,中开泵厂家,双吸中开泵-山东博二泵业有限公司 | PCB厂|线路板厂|深圳线路板厂|软硬结合板厂|电路板生产厂家|线路板|深圳电路板厂家|铝基板厂家|深联电路-专业生产PCB研发制造 | 悬浮拼装地板_幼儿园_篮球场_悬浮拼接地板-山东悬浮拼装地板厂家 | 有源电力滤波装置-电力有源滤波器-低压穿排电流互感器|安科瑞 | 基本型顶空进样器-全自动热脱附解吸仪价格-AutoHS全模式-成都科林分析技术有限公司 | 10吨无线拉力计-2吨拉力计价格-上海佳宜电子科技有限公司 | 筛分机|振动筛分机|气流筛分机|筛分机厂家-新乡市大汉振动机械有限公司 | 集菌仪厂家_全封闭_封闭式_智能智能集菌仪厂家-上海郓曹 | 广西绿桂涂料--承接隔热涂料、隔音涂料、真石漆、多彩仿石漆等涂料工程双包施工 | 成都装修公司-成都装修设计公司推荐-成都朗煜装饰公司 | 宿松新闻网 宿松网|宿松在线|宿松门户|安徽宿松(直管县)|宿松新闻综合网站|宿松官方新闻发布 | 全国国际化学校_国际高中招生_一站式升学择校服务-国际学校网 | 南京展台搭建-南京展会设计-南京展览设计公司-南京展厅展示设计-南京汇雅展览工程有限公司 | 升降炉_真空气氛炉_管式电阻炉厂家-山东中辰电炉有限公司 | 知名电动蝶阀,电动球阀,气动蝶阀,气动球阀生产厂家|价格透明-【固菲阀门官网】 | 河南中专学校|职高|技校招生-河南中职中专网 | 山东钢格板|栅格板生产厂家供应商-日照森亿钢格板有限公司 | 吉林污水处理公司,长春工业污水处理设备,净水设备-长春易洁环保科技有限公司 | 驾驶式洗地机/扫地机_全自动洗地机_工业洗地机_荣事达工厂官网 | 箱式破碎机_移动方箱式破碎机/价格/厂家_【华盛铭重工】 | 杭州成人高考_浙江省成人高考网上报名| 自动螺旋上料机厂家价格-斗式提升机定制-螺杆绞龙输送机-杰凯上料机 | 拖链电缆_柔性电缆_伺服电缆_坦克链电缆-深圳市顺电工业电缆有限公司 | 手术室净化厂家-成都做医院净化工程的公司-四川华锐-15年特殊科室建设经验 |