相同类方法之间调用,注解失效的问题
問題:
在Spring管理的項目中,方法A使用了Transactional注解,試圖實(shí)現(xiàn)事務(wù)性。但當(dāng)同一個class中的方法B調(diào)用方法A時,會發(fā)現(xiàn)方法A中的異常不再導(dǎo)致回滾,也即事務(wù)失效了。
原因:
Transactional是Spring提供的事務(wù)管理注解。
重點(diǎn)在于,Spring采用動態(tài)代理(AOP)實(shí)現(xiàn)對bean的管理和切片,它為我們的每個class生成一個代理對象。只有在代理對象之間進(jìn)行調(diào)用時,可以觸發(fā)切面邏輯。(Spring動態(tài)代理詳情可戳:spring自動加載,使用實(shí)現(xiàn)類無法加載,使用接口卻可以的原因)
而在同一個class中,方法B調(diào)用方法A,調(diào)用的是原對象的方法,而不通過代理對象。所以Spring無法切到這次調(diào)用,也就無法通過注解保證事務(wù)性了。
解決方法1:
將事務(wù)方法放到另一個類中(或者單獨(dú)開啟一層,取名“事務(wù)層”)進(jìn)行調(diào)用,即符合了在對象之間調(diào)用的條件。
解決方法2:
獲取本對象的代理對象,再進(jìn)行調(diào)用。具體操作如:
Spring-content.xml上下文中,增加配置:<aop:aspectj-autoproxy expose-proxy=“true”/>
在xxxServiceImpl中,用(xxxService)(AopContext.currentProxy()),獲取到xxxService的代理類,再調(diào)用事務(wù)方法,強(qiáng)行經(jīng)過代理類,激活事務(wù)切面。
解決方法3:
很多時候,方法內(nèi)調(diào)用又希望激活事務(wù),是由于同一個方法既有DAO操作又有I/O等耗時操作,不想讓耗時的I/O造成事務(wù)的太長耗時(比如新增商品同時需要寫入庫存)。此時,可以將I/O做成異步操作(如加入線程池),而加入線程池的操作即便加入事務(wù)也不會導(dǎo)致事務(wù)太長,問題可以迎刃而解。
創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎勵來咯,堅持創(chuàng)作打卡瓜分現(xiàn)金大獎總結(jié)
以上是生活随笔為你收集整理的相同类方法之间调用,注解失效的问题的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。