Spring中@Transactional注解的使用詳解
一、介紹
在數(shù)據(jù)庫(kù)操作中,我們經(jīng)常會(huì)遇到需要執(zhí)行多個(gè)操作,但這些操作必須是一個(gè)整體,要么全部成功,要么全部失敗。這就是事務(wù)的概念。在Spring框架中,我們可以通過(guò)@Transactional注解來(lái)聲明事務(wù)。
@Transactional注解是Spring提供的一種聲明式事務(wù)管理方式。它可以在類(lèi)或者方法級(jí)別使用,表示當(dāng)前方法或者類(lèi)中的所有方法都需要在一個(gè)事務(wù)中執(zhí)行。如果方法中出現(xiàn)異常,那么事務(wù)就會(huì)被回滾,否則就會(huì)被提交。
二、原理分析
Spring的事務(wù)管理是基于AOP(Aspect Oriented Programming,面向切面編程)的。當(dāng)你在方法或類(lèi)上使用@Transactional注解時(shí),Spring會(huì)為該方法或類(lèi)創(chuàng)建一個(gè)代理對(duì)象。當(dāng)調(diào)用被注解的方法時(shí),實(shí)際上是調(diào)用的代理對(duì)象的方法。在代理方法中,Spring會(huì)在目標(biāo)方法執(zhí)行前后添加事務(wù)管理的代碼。
Spring事務(wù)管理的核心是PlatformTransactionManager接口,它定義了事務(wù)管理的基本操作,包括事務(wù)的獲取、提交、回滾等。根據(jù)不同的數(shù)據(jù)訪問(wèn)技術(shù)(如JDBC、Hibernate、JPA等),Spring提供了不同的PlatformTransactionManager實(shí)現(xiàn)。
三、使用
使用@Transactional注解非常簡(jiǎn)單,只需在需要事務(wù)管理的方法或類(lèi)上添加此注解即可。例如:
@Service public class UserService { @Autowired private UserRepository userRepository; @Transactional public void createUser(User user) { userRepository.save(user); } }
在上述例子中,createUser
方法被@Transactional注解,這意味著如果在執(zhí)行userRepository.save(user)
過(guò)程中出現(xiàn)任何異常,那么這個(gè)操作將會(huì)被回滾,否則就會(huì)被提交。
四、本地事務(wù)與分布式事務(wù)
1.本地事務(wù):在單一的數(shù)據(jù)庫(kù)環(huán)境中,事務(wù)管理相對(duì)簡(jiǎn)單,只需確保在同一個(gè)數(shù)據(jù)庫(kù)連接上執(zhí)行的所有操作都在同一事務(wù)中即可。本地事務(wù)只涉及到一個(gè)數(shù)據(jù)庫(kù)系統(tǒng),其一致性和原子性由數(shù)據(jù)庫(kù)系統(tǒng)自己保證。
2.分布式事務(wù):隨著微服務(wù)的普及,一個(gè)業(yè)務(wù)操作可能涉及到多個(gè)服務(wù),而每個(gè)服務(wù)可能有自己的數(shù)據(jù)庫(kù)。這就需要跨多個(gè)數(shù)據(jù)庫(kù)(甚至是跨不同類(lèi)型的數(shù)據(jù)存儲(chǔ)系統(tǒng))的事務(wù)管理,也就是分布式事務(wù)。分布式事務(wù)的主要問(wèn)題是如何保證跨多個(gè)數(shù)據(jù)庫(kù)的操作的一致性和原子性。
五、 @Transactional 和分布式事務(wù)
@Transactional注解在Spring框架中主要是用于處理單一數(shù)據(jù)源的事務(wù)管理,也就是本地事務(wù)。然而,在分布式環(huán)境中,一個(gè)業(yè)務(wù)操作可能涉及到多個(gè)數(shù)據(jù)庫(kù),這就需要跨多個(gè)數(shù)據(jù)庫(kù)的事務(wù)管理,也就是分布式事務(wù)。雖然有一些復(fù)雜的方式可以實(shí)現(xiàn)分布式事務(wù),但總體來(lái)說(shuō),使用@Transactional注解來(lái)處理分布式事務(wù)是比較困難的。
如果在微服務(wù)架構(gòu)中需要使用分布式事務(wù),你可能需要使用更復(fù)雜的解決方案,比如兩階段提交協(xié)議(Two Phase Commit Protocol)或者最終一致性和補(bǔ)償機(jī)制(也被稱為Saga模式)。Spring Boot中的實(shí)現(xiàn)方式一般是通過(guò)引入一個(gè)新的組件,負(fù)責(zé)在微服務(wù)之間管理分布式事務(wù)。這個(gè)組件被稱為"transaction-server",它提供了REST API供其他微服務(wù)使用,用于添加新的事務(wù)和更新事務(wù)狀態(tài)。當(dāng)收到來(lái)自源微服務(wù)的事務(wù)確認(rèn)或回滾信息后,它會(huì)發(fā)送異步廣播事件。所有其他的微服務(wù)監(jiān)聽(tīng)這些事件,接收到事件后進(jìn)行事務(wù)的提交或回滾。
在這個(gè)模式中,分布式事務(wù)的管理并不是由@Transactional注解直接完成的,而是通過(guò)組合使用多個(gè)組件和技術(shù)來(lái)實(shí)現(xiàn)的,其中包括事務(wù)服務(wù)器、消息隊(duì)列(如RabbitMQ)、服務(wù)發(fā)現(xiàn)(如Eureka)等。
因此,雖然@Transactional注解本身不能直接處理分布式事務(wù),但它仍然是在分布式事務(wù)管理中扮演了重要角色。在設(shè)計(jì)和實(shí)現(xiàn)分布式事務(wù)的解決方案時(shí),你需要考慮到事務(wù)管理的復(fù)雜性,盡量避免在微服務(wù)架構(gòu)中使用分布式事務(wù)。
六、 微服務(wù)fegin的遠(yuǎn)程調(diào)用是否可以用@Transactional控制
在springcloud中的feign中,不能使用@Transactional注解來(lái)控制Feign的遠(yuǎn)程調(diào)用。這是因?yàn)锧Transactional是基于數(shù)據(jù)庫(kù)的事務(wù)模型來(lái)設(shè)計(jì)的,它的作用范圍是單個(gè)數(shù)據(jù)庫(kù)事務(wù)。當(dāng)你在一個(gè)被@Transactional注解的方法中調(diào)用Feign的遠(yuǎn)程服務(wù)時(shí),這個(gè)遠(yuǎn)程調(diào)用是不會(huì)被包含在當(dāng)前的數(shù)據(jù)庫(kù)事務(wù)中的。也就是說(shuō),如果你的遠(yuǎn)程調(diào)用失敗了,你的數(shù)據(jù)庫(kù)事務(wù)不會(huì)回滾。
然而,如果你的遠(yuǎn)程服務(wù)調(diào)用涉及到數(shù)據(jù)庫(kù)操作,并且你希望這個(gè)操作能在一個(gè)事務(wù)中執(zhí)行,那么你可以在你的服務(wù)端方法上使用@Transactional注解。這樣,當(dāng)你的Feign客戶端調(diào)用這個(gè)服務(wù)時(shí),服務(wù)端的數(shù)據(jù)庫(kù)操作會(huì)在一個(gè)新的事務(wù)中執(zhí)行。但是,這個(gè)事務(wù)是在服務(wù)端,而不是在調(diào)用Feign的客戶端。
總的來(lái)說(shuō),你不能使用@Transactional注解來(lái)直接控制Feign的遠(yuǎn)程調(diào)用,但你可以在服務(wù)端的方法上使用@Transactional來(lái)確保服務(wù)端的數(shù)據(jù)庫(kù)操作在一個(gè)事務(wù)中執(zhí)行。
到此這篇關(guān)于Spring中@Transactional注解的使用詳解的文章就介紹到這了,更多相關(guān)Spring @Transactional內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
idea運(yùn)行程序報(bào)錯(cuò)java程序包org.junit不存在解決辦法
這篇文章主要給大家介紹了關(guān)于idea運(yùn)行程序報(bào)錯(cuò)java程序包org.junit不存在的解決辦法, 當(dāng)出現(xiàn)程序包org.junit不存在的問(wèn)題時(shí),可以通過(guò)使用適當(dāng)?shù)腏Unit版本、添加依賴或重新下載程序包等方式進(jìn)行解決,需要的朋友可以參考下2024-02-02Java數(shù)據(jù)庫(kù)存儲(chǔ)數(shù)組的方法小結(jié)
在現(xiàn)代軟件開(kāi)發(fā)中,數(shù)組是常用的數(shù)據(jù)結(jié)構(gòu)之一,然而,在關(guān)系數(shù)據(jù)庫(kù)中直接存儲(chǔ)數(shù)組并不是一個(gè)簡(jiǎn)單的任務(wù),本文將詳細(xì)介紹幾種在Java中將數(shù)組存儲(chǔ)到數(shù)據(jù)庫(kù)的方法,包括使用JPA、JSON、XML、以及關(guān)系型數(shù)據(jù)庫(kù)的數(shù)組類(lèi)型等,需要的朋友可以參考下2024-09-09Java實(shí)現(xiàn)注冊(cè)郵箱激活賬戶實(shí)例代碼
本篇文章主要介紹了Java實(shí)現(xiàn)郵箱激活賬戶實(shí)例代碼,這里整理了詳細(xì)的代碼,具有一定的參考價(jià)值,有需要的小伙伴可以參考下。2017-07-07關(guān)于IDEA使用jsp可以訪問(wèn)頁(yè)面轉(zhuǎn)換為html彈出頁(yè)面為404的問(wèn)題
這篇文章主要介紹了關(guān)于IDEA使用jsp可以訪問(wèn)頁(yè)面轉(zhuǎn)換為html彈出頁(yè)面為404的問(wèn)題及解決方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-12-12IntelliJ?IDEA2022.3?springboot?熱部署含靜態(tài)文件(最新推薦)
這篇文章主要介紹了IntelliJ?IDEA2022.3?springboot?熱部署含靜態(tài)文件,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-01-01