SpringBoot中的聲明式事務(wù)+切面事務(wù)+編程式事務(wù)詳解
前言
事務(wù)管理對(duì)于企業(yè)應(yīng)用來說是至關(guān)重要的,當(dāng)出現(xiàn)異常情況時(shí),它也可以保證數(shù)據(jù)的一致性。
springBoot中兩種事務(wù)的實(shí)現(xiàn)方式,編程式事務(wù)配置和聲明式事務(wù)配置還有切面事務(wù) 還有以后的分布式事務(wù)
一、事務(wù)特性
- 原子性(Atomicity):
- 事務(wù)是一個(gè)原子操作,由一系列動(dòng)作組成。事務(wù)的原子性確保動(dòng)作要么全部完成,要么完全不起作用;
- 一致性(Consistency):
- 一旦事務(wù)完成(不管是成功還是失?。?,系統(tǒng)必須確保它所建模的業(yè)務(wù)處于一致的狀態(tài),而不會(huì)是部分完成部分失敗。在現(xiàn)實(shí)中的數(shù)據(jù)不應(yīng)該被破壞;
- 隔離性(Isolation):
- 可能有許多事務(wù)會(huì)同時(shí)處理相同的數(shù)據(jù),因此每個(gè)事務(wù)都應(yīng)該與其他事務(wù)隔離開來,防止數(shù)據(jù)損壞;
- 持久性(Durability):
- 一旦事務(wù)完成,無論發(fā)生什么系統(tǒng)錯(cuò)誤,它的結(jié)果都不應(yīng)該受到影響,這樣就能從任何系統(tǒng)崩潰中恢復(fù)過來。通常情況下,事務(wù)的結(jié)果被寫到持久化存儲(chǔ)器中;
Spring 框架中,涉及到事務(wù)管理的 API 大約有100個(gè)左右,其中最重要的有三個(gè): TransactionDefinition、PlatformTransactionManager、TransactionStatus
。
所謂事務(wù)管理,其實(shí)就是”按照給定的事務(wù)規(guī)則來執(zhí)行提交或者回滾操作”。”
給定的事務(wù)規(guī)則”就是用 TransactionDefinition 表示的,”按照……來執(zhí)行提交或者回滾操作”便是用 PlatformTransactionManager 來表示
而 TransactionStatus 用于表示一個(gè)運(yùn)行著的事務(wù)的狀態(tài)。
打一個(gè)不恰當(dāng)?shù)谋扔?,TransactionDefinition 與 TransactionStatus 的關(guān)系就像程序和進(jìn)程的關(guān)系。
開啟事務(wù) @EnableTransactionManagement
二、事務(wù)的隔離級(jí)別
和數(shù)據(jù)庫中的事務(wù)級(jí)別是一樣的 都會(huì)出現(xiàn)事務(wù)本身應(yīng)該有的問題
三、事務(wù)的傳播行為
四、 Springboot事務(wù)
1.Springboot聲明式事務(wù)
聲明式事務(wù)是建立在AOP之上的。
其本質(zhì)是對(duì)方法前后進(jìn)行攔截,然后在目標(biāo)方法開始之前創(chuàng)建或者加入一個(gè)事務(wù),在執(zhí)行完目標(biāo)方法之后根據(jù)執(zhí)行情況提交或者回滾事務(wù)。
Spring配置文件中關(guān)于事務(wù)配置總是由三個(gè)組成部分,分別是DataSource、TransactionManager和代理機(jī)制這三部分,無論哪種配置方式,一般變化的只是代理機(jī)制這部分。
DataSource、TransactionManager這兩部分只是會(huì)根據(jù)數(shù)據(jù)訪問方式有所變化,比如使用Hibernate進(jìn)行數(shù)據(jù)訪問時(shí),DataSource實(shí)際為SessionFactory,TransactionManager的實(shí)現(xiàn)為HibernateTransactionManager。
根據(jù)代理機(jī)制的不同,總結(jié)了五種Spring事務(wù)的配置方式,如下圖:
聲明式事務(wù)@Transactional可以使用在類上,也可以使用在public方法上
如果是使用在類上,則是對(duì)所有的public方法都開啟事務(wù),如果類和方法上都有則方法上的事務(wù)生效
在類上
@Transactional(rollbackFor=Exception.class) public class TransactionServiceImpl implements TransactionService { }
在方法上
@Override @Transactional(rollbackFor=Exception.class) public void t1(Student one) { }
聲明式事務(wù)最大的優(yōu)點(diǎn)就是不需要通過編程的方式管理事務(wù),這樣就不需要在業(yè)務(wù)邏輯代碼中摻雜事務(wù)管理的代碼,只需在配置文件中做相關(guān)的事務(wù)規(guī)則聲明(或通過基于@Transactional注解的方式),便可以將事務(wù)規(guī)則應(yīng)用到業(yè)務(wù)邏輯中。
優(yōu)點(diǎn):
編程式事務(wù)每次實(shí)現(xiàn)都要單獨(dú)實(shí)現(xiàn),但業(yè)務(wù)量大且功能復(fù)雜時(shí),使用編程性事務(wù)無疑是痛苦的;而聲明式事務(wù)不同,聲明式事務(wù)屬于非侵入性,不會(huì)影響業(yè)務(wù)邏輯的實(shí)現(xiàn),只需在配置文件中做相關(guān)的事務(wù)規(guī)則聲明(或通過基于@Transactional注解的方式),便可以將事務(wù)規(guī)則應(yīng)用到業(yè)務(wù)邏輯中;
非侵入式的開發(fā)方式,聲明式事務(wù)管理使業(yè)務(wù)代碼不受污染,一個(gè)普通的POJO對(duì)象,只要加上注解就可以獲得完全的事務(wù)支持;
缺點(diǎn):
最細(xì)粒度只能是作用到方法級(jí)別,無法做到像編程事務(wù)那樣可以作用到代碼塊級(jí)別;
實(shí)現(xiàn)方式:
使用攔截器:基于TransactionInterceptor 類來實(shí)施聲明式事務(wù)管理功能(Spring最初提供的實(shí)現(xiàn)方式);
Bean和代理:基于 TransactionProxyFactoryBean的聲明式事務(wù)管理
使用tx標(biāo)簽配置的攔截器:基于tx和aop名字空間的xml配置文件(基于Aspectj AOP配置事務(wù)); 全注解:基于@Transactional注解;
@Transactional的參數(shù)
聲明式事務(wù)的約定流程:
首先Spring通過事務(wù)管理器(PlatformTransactionManager的子類)創(chuàng)建事務(wù),與此同時(shí)會(huì)把事務(wù)定義中的隔離級(jí)別、超時(shí)時(shí)間等屬性根據(jù)配置內(nèi)容往事務(wù)上設(shè)置。
而根據(jù)傳播行為配置采取一種特定的策略,后面會(huì)談到傳播行為的使用問題,這是Spring根據(jù)配置完成的內(nèi)容,你只需要配置,無須編碼。
然后,啟動(dòng)開發(fā)者提供的業(yè)務(wù)代碼,我們知道Spring會(huì)通過反射的方式調(diào)度開發(fā)者的業(yè)務(wù)代碼,但是反射的結(jié)果可能是正常返回或者產(chǎn)生異常返回,那么它給的約定是只要發(fā)生異常,并且符合事務(wù)定義類回滾條件的,Spring就會(huì)將數(shù)據(jù)庫事務(wù)回滾,否則將數(shù)據(jù)庫事務(wù)提交,這也是Spring自己完成的
顯然聲明式事務(wù)管理要優(yōu)于編程式事務(wù)管理,這正是spring倡導(dǎo)的非侵入式的開發(fā)方式。
聲明式事務(wù)管理使業(yè)務(wù)代碼不受污染,一個(gè)普通的POJO對(duì)象,只要加上注解就可以獲得完全的事務(wù)支持。和編程式事務(wù)相比,聲明式事務(wù)唯一不足地方是,它的最細(xì)粒度只能作用到方法級(jí)別,無法做到像編程式事務(wù)那樣可以作用到代碼塊級(jí)別。
但是即便有這樣的需求,也存在很多變通的方法,比如,可以將需要進(jìn)行事務(wù)管理的代碼塊獨(dú)立為方法等等。
聲明式事務(wù)管理也有兩種常用的方式,一種是基于tx和aop名字空間的xml配置文件,另一種就是基于@Transactional注解。
顯然基于注解的方式更簡單易用,更清爽。
2. Springboot編程式事務(wù)
編程式事務(wù): 是侵入性事務(wù)管理,直接使用底層的PlatformTransactionManager、使用TransactionTemplate(Spring推薦使用);
編程式事務(wù)管理對(duì)基于 POJO 的應(yīng)用來說是唯一選擇。
我們需要在代碼中調(diào)用beginTransaction()、commit()、rollback()等事務(wù)管理相關(guān)的方法;
@Autowired private TransactionTemplate transactionTemplate;
@Override public final void save2() { transactionTemplate.execute((status)->{ mapper.saveStudent(newOne()); mapper.saveStudent(newOne()); return Boolean.TRUE; }); }
這樣兩個(gè)mapper.saveStudent(newOne());就在一個(gè)事務(wù)中執(zhí)行了
3.SpringBoo切面編程式事務(wù)
SpringBoo切面編程式事務(wù)應(yīng)該是聲明式事務(wù)的一種具體體現(xiàn) 此種方式基于AOP功能,所以需要添加
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency>
配置類
@Aspect @Configuration public class MyTransactionConfig { /** * 配置方法過期時(shí)間,默認(rèn)-1,永不超時(shí) */ private final static int TX_METHOD_TIME_OUT = 10; /** * 全局事務(wù)位置配置 在哪些地方需要進(jìn)行事務(wù)處理 * 配置切入點(diǎn)表達(dá)式 */ private static final String POITCUT_EXPRESSION = "execution(* zdc.enterprise.service.impl.*.*(..))"; @Autowired private PlatformTransactionManager platformTransactionManager; @Bean public TransactionInterceptor txadvice() { /*只讀事物、不做更新刪除等*/ /*事務(wù)管理規(guī)則*/ RuleBasedTransactionAttribute readOnlyRule = new RuleBasedTransactionAttribute(); /*設(shè)置當(dāng)前事務(wù)是否為只讀事務(wù),true為只讀*/ readOnlyRule.setReadOnly(true); /* transactiondefinition 定義事務(wù)的隔離級(jí)別; *如果當(dāng)前存在事務(wù),則加入該事務(wù);如果當(dāng)前沒有事務(wù),則以非事務(wù)的方式繼續(xù)運(yùn)行。*/ readOnlyRule.setPropagationBehavior(TransactionDefinition.PROPAGATION_SUPPORTS); /*增刪改事務(wù)規(guī)則*/ RuleBasedTransactionAttribute requireRule = new RuleBasedTransactionAttribute(); /*拋出異常后執(zhí)行切點(diǎn)回滾 建議自定義異常*/ requireRule.setRollbackRules(Collections.singletonList(new RollbackRuleAttribute(Exception.class))); /*PROPAGATION_REQUIRED:事務(wù)隔離性為1,若當(dāng)前存在事務(wù),則加入該事務(wù);如果當(dāng)前沒有事務(wù),則創(chuàng)建一個(gè)新的事務(wù)。這是默認(rèn)值。 */ requireRule.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED); /*設(shè)置事務(wù)失效時(shí)間,超過10秒*/ requireRule.setTimeout(TX_METHOD_TIME_OUT); /** 配置事務(wù)管理規(guī)則 nameMap聲明具備需要管理事務(wù)的方法名. 這里使用addTransactionalMethod 使用setNameMap */ Map<String, TransactionAttribute> nameMap = new HashMap<>(); nameMap.put("add*", requireRule); nameMap.put("save*", requireRule); nameMap.put("insert*", requireRule); nameMap.put("update*", requireRule); nameMap.put("delete*", requireRule); nameMap.put("remove*", requireRule); /*進(jìn)行批量操作時(shí)*/ nameMap.put("batch*", requireRule); nameMap.put("get*", readOnlyRule); nameMap.put("query*", readOnlyRule); nameMap.put("find*", readOnlyRule); nameMap.put("select*", readOnlyRule); nameMap.put("count*", readOnlyRule); NameMatchTransactionAttributeSource source = new NameMatchTransactionAttributeSource(); source.setNameMap(nameMap); TransactionInterceptor transactionInterceptor = new TransactionInterceptor(platformTransactionManager, source); return transactionInterceptor; } /** * 設(shè)置切面=切點(diǎn)pointcut+通知TxAdvice * @return */ @Bean public Advisor txAdviceAdvisor() { AspectJExpressionPointcut pointcut = new AspectJExpressionPointcut(); pointcut.setExpression(POITCUT_EXPRESSION); return new DefaultPointcutAdvisor(pointcut, txadvice()); } }
有了這個(gè)切面配置類,就不要用在類或者每個(gè)方法上使用@Transactional了,當(dāng)然方法名前綴要能和設(shè)置的匹配上
RuleBasedTransactionAttribute的參數(shù)大致和@Transactional的參數(shù)相同,里面有詳細(xì)的注釋,就不過多解釋了
到此這篇關(guān)于SpringBoot中的聲明式事務(wù)+切面事務(wù)+編程式事務(wù)詳解的文章就介紹到這了,更多相關(guān)SpringBoot各類型事務(wù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MyBatis-Plus中通用枚舉的實(shí)現(xiàn)
表中的有些字段值是固定的此時(shí)我們可以使用MyBatis-Plus的通用枚舉來實(shí)現(xiàn),本文主要介紹了MyBatis-Plus中通用枚舉的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2024-05-05Java多線程中線程的兩種創(chuàng)建方式及比較代碼示例
這篇文章主要介紹了Java多線程中線程的兩種創(chuàng)建方式及比較代碼示例,簡單介紹了線程的概念,并行與并發(fā)等,然后通過實(shí)例代碼向大家展示了線程的創(chuàng)建,具有一定參考價(jià)值,需要的朋友可以了解下。2017-11-11解決springboot啟動(dòng)成功,但訪問404的問題
這篇文章主要介紹了解決springboot啟動(dòng)成功,但訪問404的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-07-07Springboot中路徑參數(shù)帶 (%2F)的問題徹底解決方案
這篇文章主要介紹了徹底解決Springboot中路徑參數(shù)帶(%2F)的問題,本文結(jié)合示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-06-06logback StatusListener的定義方法源碼解讀
這篇文章主要為大家介紹了logback StatusListener的定義方法源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11