Spring aop 如何通過獲取代理對象實(shí)現(xiàn)事務(wù)切換
Spring aop 獲取代理對象實(shí)現(xiàn)事務(wù)切換
在項(xiàng)目中,涉及到同一個類中一個方法調(diào)用另外一個方法,并且兩個方法的事務(wù)不相關(guān),
這里面涉及到一個事務(wù)切換的問題,一般的方法沒問題,根據(jù)通過aop注解在方法上通過加注解標(biāo)識,
答案是:
通過spring aop類里面的AopContext類獲取當(dāng)前類的代理對象,
這樣就能切換對應(yīng)的事務(wù)管理器了,具體做法如下:
(1).在applicationContext.xml文件中配置如下:
<!-- 開啟暴露Aop代理到ThreadLocal支持 --> <aop:aspectj-autoproxy expose-proxy="true"/>
(2).在需要切換的地方獲取代理對象,
再調(diào)用對應(yīng)的方法,如下:
((類名) AopContext.currentProxy()).方法();
(3).注意
這里需要被代理對象使用的方法必須是public類型的方法,不然獲取不到代理對象,會報下面的錯誤:
java.lang.IllegalStateException: Cannot find current proxy: Set 'exposeProxy' property on Advised to 'true' to make it available.
開啟暴露AOP代理即可.
因?yàn)殚_啟事務(wù)和事務(wù)回滾,實(shí)際這個過程是aop代理幫忙完成的,當(dāng)調(diào)用一個方法時,它會先檢查時候有事務(wù),有則開啟事務(wù),
當(dāng)調(diào)用本類的方法是,它并沒有將其視為proxy調(diào)用,而是方法的直接調(diào)用,所以也就沒有檢查該方法是否含有事務(wù)這個過程,
那么本地方法調(diào)用的事務(wù)也就無效了。
獲取代理bean的原始對象
public class AopTargetUtil { /** * 獲取 目標(biāo)對象 * @param proxy 代理對象 * @return * @throws Exception */ public static Object getTarget(Object proxy) throws Exception { if(!AopUtils.isAopProxy(proxy)) { return proxy;//不是代理對象 } 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; } }
以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Idea中Java項(xiàng)目如何修改項(xiàng)目名
這篇文章主要介紹了Idea中Java項(xiàng)目如何修改項(xiàng)目名問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-06-06Java并發(fā)編程加鎖導(dǎo)致的活躍性問題詳解方案
所謂并發(fā)編程是指在一臺處理器上"同時"處理多個任務(wù)。并發(fā)是在同一實(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)文件上傳、下載,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-10-10SpringBoot引入模板引擎實(shí)現(xiàn)視圖解析
這篇文章主要介紹了SpringBoot引入模板引擎實(shí)現(xiàn)視圖解析方法流程,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2022-10-10