java - Spring事務(wù)配置在service層,傳播規(guī)則為required,方法中究竟應(yīng)該是調(diào)用service還是多個dao比較好?
問題描述
Spring中事務(wù)配置如下:
<tx:advice transaction-manager='transactionManager'> <tx:attributes><tx:method name='delete*' propagation='REQUIRED' read-only='false' rollback-for='java.lang.Exception'/><tx:method name='insert*' propagation='REQUIRED' read-only='false' rollback-for='Exception' /><tx:method name='update*' propagation='REQUIRED' read-only='false' rollback-for='java.lang.Exception' /><tx:method name='save*' propagation='REQUIRED' read-only='false' rollback-for='Exception' /><tx:method name='*' propagation='REQUIRED' read-only='true'/> <:attributes><:advice>
現(xiàn)在ServiceA中有一個方法methodA,那么在ServiceA中應(yīng)該注入ServiceB,ServiceC呢,還是DaoB,DaoC,然后在methodA中去保存B,C,保證B,C同時保存成功,或同時失敗!
答:
既可以單獨注入service,也可以單獨注入dao,關(guān)鍵是,spring容器的事務(wù)管理默認(rèn)只截獲未檢查異常RuntimeException。上邊配置的rollback-for='java.lang.Exception'其實不用配置。配置如下
<tx:advice transaction-manager='transactionManager'> <tx:attributes><tx:method name='delete*' propagation='REQUIRED' read-only='false' /><tx:method name='insert*' propagation='REQUIRED' read-only='false' /><tx:method name='update*' propagation='REQUIRED' read-only='false' /><tx:method name='save*' propagation='REQUIRED' read-only='false' /><tx:method name='*' propagation='REQUIRED' read-only='true'/> <:attributes><:advice>
解決方案是:
如果代碼中使用了try...catch...捕獲了檢查型異常,意味著程序員自己必須要解決異常,必須知道如何解決異常。通常的做法是:將檢查型的異常在catch塊中重新拋出為Runtime Exception,這樣Spring容器就會截獲該異常,進行事務(wù)回滾處理 。如下
try { .....}catch( CheckedException e ) { logger.error(e); throw new RuntimeException(e);}
注意,不使用try...catch...,而在方法簽名后向外拋出檢查型異常的行為不可取,事務(wù)也不會回滾。
如果代碼中沒有使用try拋出了未檢查異常,則Spring容器會自動截獲異常,進行事務(wù)回滾處理。
問題解答
回答1:如果你想更多了解Spring事務(wù)機制可以看我的這幾篇文章:
Spring Transaction詳解 - Transaction Isolation
Spring Transaction詳解 - Transaction Propagation模式
Spring Transaction詳解 - 手動回滾事務(wù)
Spring Transaction詳解 - 異常發(fā)生時的事務(wù)回滾機制
回答2:其實這種事情就是根據(jù)需要了,事務(wù)是會自動合并的,但作為設(shè)計考慮,盡量調(diào)用 dao 這樣能夠使不同的 service 得以解偶。
回答3:一般我們在Service的方法上會進行事務(wù)的定義,特別是如果有控制傳播行為的場景,那放入dao就和放入service不同了。因為dao肯定都是在一個大事務(wù)下了,service就比較復(fù)雜了。
