作者:Mazin
https://my.oschina.net/u/3441184/blog/893628

這樣的情景可能不常見,但是還是會有的,一旦遇到,如果業務比較復雜,就會很麻煩,但是還是有解決的方案的,比如將一個service方法拆成兩個方法,也就是將兩個操作的事務分開,
但是這只適用與業務比較簡單的,如果出現多次資料庫的寫操作,而我們呼叫的系統只需要其中一個寫操作的最新資料,如果我們將它分開,那么如果呼叫目標系統出現例外的時候,那么之前的寫操作就不能回滾了,
舉個簡單的例子:
@Service
public class ServiceA {
@Transactional
public void doSomething(){
向資料庫中添加資料;
呼叫其他系統;
}
}
這里就用偽代碼來做示例了,當我們執行了“向資料庫中添加資料”,我們去資料庫中查詢,發現并沒有我們添加的資料,但是當我們的service這個方法執行完成之后,資料庫中就有這條資料了,這是由于資料庫的隔離性造成的,
Spring中的事務注解 @transactional 提供了一個引數:
Propagation propagation() default Propagation.REQUIRED;
這個引數是定義 Spring事務的傳遞性的,默認值為:required,也就是如果有事務,就加入事務,如果沒有,就創建事務,
這個引數的值有很多,例如:REQUIRES_NEW,這個值就代表創建一個新的事務,與原來的事務分開,這個好像能解決我們的問題,關注Java技術堆疊公眾號在后臺回復:spring,可獲取一份堆疊長整理的最新 Spring 系列技術干貨,
我們將剛剛那個方法修改一下:
@Service
public class ServiceA {
@Transactional
public void doSomething(){
insert();
呼叫其他系統;
}
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void insert(){
向資料庫中添加資料;
}
}
執行之后,發現結果還是沒有改變,必須要整體執行完成,資料庫中資料才會出現,說明還是在一個事務中,Spring的核心思想,推薦你看看,
我們再將代碼修改一下:
@Service
public class ServiceA {
@Autowired
private ServiceB serviceB;
@Transactional
public void doSomething(){
serviceB.insert();
呼叫其他系統;
}
}
@Service
public class ServiceB {
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void insert(){
向資料庫中添加資料;
}
}
我們將要事務分離出來的方法寫在另一個service中,再次測驗,發現執行完插入陳述句之后,資料庫中就已經能查到資料了,說明事務分離了,完成了我們的需求,
當然** Spring**其實也考慮這個,在 Spring的配置中,我們只需要添加標簽:
aop:aspectj-autoproxy expose-proxy="true"/
或者:
aop:config expose-proxy="true"
并且在代碼的呼叫中要求使用代理物件去呼叫即可:
((ServiceA ) AopContext.currentProxy()).insert();
關注公眾號Java技術堆疊回復"面試"獲取我整理的2020最全面試題及答案,
推薦去我的博客閱讀更多:
1.Java JVM、集合、多執行緒、新特性系列教程
2.Spring MVC、Spring Boot、Spring Cloud 系列教程
3.Maven、Git、Eclipse、Intellij IDEA 系列工具教程
4.Java、后端、架構、阿里巴巴等大廠最新面試題
覺得不錯,別忘了點贊+轉發哦!
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/150249.html
標籤:Java
