您好,登錄后才能下訂單哦!
這篇文章主要介紹“spring事務失效的原因有哪些”,在日常操作中,相信很多人在spring事務失效的原因有哪些問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”spring事務失效的原因有哪些”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!
Spring 事務的本質其實就是數據庫對事務的支持,沒有數據庫的事務支持,spring 是無法提供事務功能的。真正的數據庫層的事務提交和回滾是在binlog提交之后進行提交的 通過 redo log 來重做, undo log來回滾。
一般我們在程序里面使用的都是在方法上面加@Transactional
注解,這種屬于聲明式事務。
聲明式事務本質是通過 AOP 功能,對方法前后進行攔截,將事務處理的功能編織到攔截的方法中,也就是在目標方法開始之前加入一個事務,在執行完目標方法之后根據執行情況提交或者回滾事務。
這里以 MySQL 為例,其 MyISAM 引擎是不支持事務操作的,InnoDB 才是支持事務的引擎,一般要支持事務都會使用 InnoDB
@Service public class UserServiceImpl implements UserService { public void update(User user) { updateUser(user); } @Transactional(rollbackFor = Exception.class) public void updateUser(User user) { // update user } }
上面的這種情況下是不會有事務管理操作的。
通過看聲明式事務的原理可知,spring使用的是AOP切面的方式,本質上使用的是動態代理來達到事務管理的目的,當前類調用的方法上面加@Transactional
這個是沒有任何作用的,因為調用這個方法的是this
.
OK, 我們在看下面的一種例子。
@Service public class UserServiceImpl implements UserService { @Transactional(rollbackFor = Exception.class) public void update(User user) { updateUser(user); } @Transactional(propagation = Propagation.REQUIRES_NEW) public void updateUser(User user) { // update user } }
這次在 update 方法上加了 @Transactional
,updateUser 加了 REQUIRES_NEW
新開啟一個事務,那么新開的事務管用么?
答案是:不管用!
因為它們發生了自身調用,就調該類自己的方法,而沒有經過 Spring 的代理類,默認只有在外部調用事務才會生效,這也是老生常談的經典問題了。
@Service public class UserServiceImpl implements UserService { @Transactional(rollbackFor = Exception.class) private void updateUser(User user) { // update user } }
private
方法是不會被spring代理的,因此是不會有事務產生的,這種做法是無效的。
//@Service public class UserServiceImpl implements UserService { @Transactional(rollbackFor = Exception.class) public void updateUser(User user) { // update user } }
沒有被spring管理的bean, spring連代理對象都無法生成,當然無效咯。
@Service public class UserServiceImpl implements UserService { @Transactional(propagation = Propagation.NOT_SUPPORTED) public void update(User user) { // update user } }
回顧一下spring的事務傳播行為
Spring 事務的傳播行為說的是,當多個事務同時存在的時候, Spring 如何處理這些事務的行為。
PROPAGATION_REQUIRED:如果當前沒有事務,就創建一個新事務,如果當前存在事務,就加入該事務,該設置是最常用的設置。
PROPAGATION_SUPPORTS:支持當前事務,如果當前存在事務,就加入該事務,如果當前不存在事務,就以非事務執行
PROPAGATION_MANDATORY:支持當前事務,如果當前存在事務,就加入該事務,如果當前不存在事務,就拋出異常。
PROPAGATION_REQUIRES_NEW:創建新事務,無論當前存不存在事務,都創建新事務。
PROPAGATION_NOT_SUPPORTED:以非事務方式執行操作,如果當前存在事務,就把當前事務掛起。
PROPAGATION_NEVER: 以非事務方式執行,如果當前存在事務,則拋出異常。
PROPAGATION_NESTED:如果當前存在事務,則在嵌套事務內執行。如果當前沒有事務,則按 REQUIRED 屬性執行
當傳播行為設置了PROPAGATION_NOT_SUPPORTED,PROPAGATION_NEVER,PROPAGATION_SUPPORTS這三種時,就有可能存在事務不生效
@Service public class UserServiceImpl implements UserService { @Transactional(rollbackFor = Exception.class) public void update(User user) { try{ // update user }catch(Execption e){ log.error("異常",e) } } }
異常被抓了,這樣子代理類就沒辦法知道你到底有沒有錯誤,需不需要回滾,所以這種情況也是沒辦法回滾的哦。
public interface UserService { @Transactional(rollbackFor = Exception.class) public void update(User user) }
@Service public class UserServiceImpl implements UserService { public void update(User user) { // update user } }
通過元素的 "proxy-target-class" 屬性值來控制是基于接口的還是基于類的代理被創建。如果 "proxy-target-class" 屬值被設置為 "true",那么基于類的代理將起作用(這時需要CGLIB庫cglib.jar在CLASSPATH中)。如果 "proxy-target-class" 屬值被設置為 "false" 或者這個屬性被省略,那么標準的JDK基于接口的代理將起作用
注解@Transactional cglib與java動態代理最大區別是代理目標對象不用實現接口,那么注解要是寫到接口方法上,要是使用cglib代理,這是注解事務就失效了,為了保持兼容注解最好都寫到實現類方法上。
@Service public class UserServiceImpl implements UserService { @Transactional public void update(User user) { // update user } }
到此,關于“spring事務失效的原因有哪些”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。