Spring aop 如何通過(guò)獲取代理對(duì)象實(shí)現(xiàn)事務(wù)切換
Spring aop 獲取代理對(duì)象實(shí)現(xiàn)事務(wù)切換
在項(xiàng)目中,涉及到同一個(gè)類中一個(gè)方法調(diào)用另外一個(gè)方法,并且兩個(gè)方法的事務(wù)不相關(guān),
這里面涉及到一個(gè)事務(wù)切換的問(wèn)題,一般的方法沒問(wèn)題,根據(jù)通過(guò)aop注解在方法上通過(guò)加注解標(biāo)識(shí),
答案是:
通過(guò)spring aop類里面的AopContext類獲取當(dāng)前類的代理對(duì)象,
這樣就能切換對(duì)應(yīng)的事務(wù)管理器了,具體做法如下:
(1).在applicationContext.xml文件中配置如下:
<!-- 開啟暴露Aop代理到ThreadLocal支持 --> <aop:aspectj-autoproxy expose-proxy="true"/>
(2).在需要切換的地方獲取代理對(duì)象,
再調(diào)用對(duì)應(yīng)的方法,如下:
((類名) AopContext.currentProxy()).方法();
(3).注意
這里需要被代理對(duì)象使用的方法必須是public類型的方法,不然獲取不到代理對(duì)象,會(huì)報(bào)下面的錯(cuò)誤:
java.lang.IllegalStateException: Cannot find current proxy: Set 'exposeProxy' property on Advised to 'true' to make it available.
開啟暴露AOP代理即可.
因?yàn)殚_啟事務(wù)和事務(wù)回滾,實(shí)際這個(gè)過(guò)程是aop代理幫忙完成的,當(dāng)調(diào)用一個(gè)方法時(shí),它會(huì)先檢查時(shí)候有事務(wù),有則開啟事務(wù),
當(dāng)調(diào)用本類的方法是,它并沒有將其視為proxy調(diào)用,而是方法的直接調(diào)用,所以也就沒有檢查該方法是否含有事務(wù)這個(gè)過(guò)程,
那么本地方法調(diào)用的事務(wù)也就無(wú)效了。
獲取代理bean的原始對(duì)象
public class AopTargetUtil { /** * 獲取 目標(biāo)對(duì)象 * @param proxy 代理對(duì)象 * @return * @throws Exception */ public static Object getTarget(Object proxy) throws Exception { if(!AopUtils.isAopProxy(proxy)) { return proxy;//不是代理對(duì)象 } if(AopUtils.isJdkDynamicProxy(proxy)) { return getJdkDynamicProxyTargetObject(proxy); } else { //cglib return getCglibProxyTargetObject(proxy); } } private static Object getCglibProxyTargetObject(Object proxy) throws Exception { Field h = proxy.getClass().getDeclaredField("CGLIB$CALLBACK_0"); h.setAccessible(true); Object dynamicAdvisedInterceptor = h.get(proxy); Field advised = dynamicAdvisedInterceptor.getClass().getDeclaredField("advised"); advised.setAccessible(true); Object target = ((AdvisedSupport)advised.get(dynamicAdvisedInterceptor)).getTargetSource().getTarget(); return target; } private static Object getJdkDynamicProxyTargetObject(Object proxy) throws Exception { Field h = proxy.getClass().getSuperclass().getDeclaredField("h"); h.setAccessible(true); AopProxy aopProxy = (AopProxy) h.get(proxy); Field advised = aopProxy.getClass().getDeclaredField("advised"); advised.setAccessible(true); Object target = ((AdvisedSupport)advised.get(aopProxy)).getTargetSource().getTarget(); return target; } }
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java?GUI實(shí)現(xiàn)多個(gè)窗口切換效果
這篇文章主要為大家詳細(xì)介紹了Java?GUI實(shí)現(xiàn)多個(gè)窗口的切換效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04Junit測(cè)試多線程無(wú)法得到結(jié)果的問(wèn)題解決
在測(cè)試一個(gè)文件轉(zhuǎn)換工具類的時(shí)候,發(fā)生一個(gè)有趣的現(xiàn)象,同樣的輸入,使用Main函數(shù)可以正確解析,得到結(jié)果,使用Junit卻無(wú)法得到結(jié)果,神奇的是,即使捕獲Throwable,也無(wú)法捕獲到仍和異常。2021-05-05Idea中Java項(xiàng)目如何修改項(xiàng)目名
這篇文章主要介紹了Idea中Java項(xiàng)目如何修改項(xiàng)目名問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-06-06Java并發(fā)編程加鎖導(dǎo)致的活躍性問(wèn)題詳解方案
所謂并發(fā)編程是指在一臺(tái)處理器上"同時(shí)"處理多個(gè)任務(wù)。并發(fā)是在同一實(shí)體上的多個(gè)事件。多個(gè)事件在同一時(shí)間間隔發(fā)生,所以編寫正確的程序很難,而編寫正確的并發(fā)程序則難上加難2021-10-10Java實(shí)現(xiàn)求子數(shù)組和的最大值算法示例
這篇文章主要介紹了Java實(shí)現(xiàn)求子數(shù)組和的最大值算法,涉及Java數(shù)組遍歷、判斷、運(yùn)算等相關(guān)操作技巧,需要的朋友可以參考下2018-02-02JavaEE組件commons-fileupload實(shí)現(xiàn)文件上傳、下載
這篇文章主要介紹了JavaEE組件commons-fileupload實(shí)現(xiàn)文件上傳、下載,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-10-10SpringBoot引入模板引擎實(shí)現(xiàn)視圖解析
這篇文章主要介紹了SpringBoot引入模板引擎實(shí)現(xiàn)視圖解析方法流程,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧2022-10-10