事務(wù),是描述一組操作的抽象,比如對(duì)數(shù)據(jù)庫(kù)的一組操作,要么全部成功,要么全部失敗。事務(wù)具有4個(gè)特性:Atomicity(原子性),Consistency(一致性),Isolation(隔離性),Durability(持久性)。在實(shí)際開(kāi)發(fā)中,我們對(duì)事務(wù)應(yīng)用最多就是在數(shù)據(jù)庫(kù)操作這一環(huán),特別是Spring對(duì)數(shù)據(jù)庫(kù)事務(wù)進(jìn)行了封裝管理。Spring對(duì)事務(wù)的支持,確實(shí)很強(qiáng)大,但是從本質(zhì)上來(lái)講:事務(wù)是否生效取決數(shù)據(jù)庫(kù)底層是否支持(比如MySQL的MyISAM引擎就不支持事務(wù),Spring能奈何?。?,同時(shí)一個(gè)事務(wù)的多個(gè)操作需要在同一個(gè)Connection上。事務(wù)也往往是在業(yè)務(wù)邏輯層來(lái)控制。本篇博客將通過(guò)手寫(xiě)一個(gè)Demo來(lái)分析Spring事務(wù)底層到底是如何幫助我們輕松完成事務(wù)管理的! 先來(lái)看一眼工程結(jié)構(gòu): 工程結(jié)構(gòu) ConnectionHolder 在Spring中,有時(shí)候我們是不是要配置多個(gè)數(shù)據(jù)源DataSource?很顯然,Spring需要通過(guò)DataSource來(lái)得到操作數(shù)據(jù)庫(kù)的管道Connection,這有點(diǎn)類似于JNDI查找。 這里通過(guò)ConnectionHolder類來(lái)完成這個(gè)過(guò)程,需要思考的是在多線程下,這顯然是存在問(wèn)題的。為避免多線程問(wèn)題,難道我們采用線程安全的Map,比如ConcurrentHashMap,其實(shí)我們真正的目的是什么?是保證一個(gè)線程下,一個(gè)事務(wù)的多個(gè)操作拿到的是一個(gè)Connection,顯然使用ConcurrentHashMap根本無(wú)法保證! Spring很聰明,她提供了一種思路,來(lái)解決,看下面的代碼! SingleThreadConnectionHolder 本來(lái)線程不安全的,通過(guò)ThreadaLocal這么封裝一下,立刻就變成了線程的局部變量,不僅僅安全了,還保證了一個(gè)線程下面的操作拿到的Connection是同一個(gè)對(duì)象!這種思想,確實(shí)非常巧妙,這也是無(wú)鎖編程思想的一種方式! TransactionManager TransactionManager,這個(gè)我們經(jīng)常在Spring里面進(jìn)行配置吧,事務(wù)大管家! UserAccountDao UserOrderDao 這里通過(guò)這2個(gè)DAO,想模擬一個(gè)事務(wù)中賬戶購(gòu)買、下單2個(gè)操作。 UserService 到這里,可以清晰的看到Spring事務(wù)管理的一個(gè)縮影了吧! 測(cè)試 這里,主要是模擬Spring的注入以及多用戶并發(fā)請(qǐng)求。 你可以發(fā)現(xiàn),一個(gè)線程中的一個(gè)事務(wù)的多個(gè)操作,使用的是同一個(gè)Connection! |
|