Spring 事務(wù) @Transactional注解 面試及原理
1. 你在項目中是如何使用事物的?
我們項目的框架都是使用的Spring,spring分為 編程式事務(wù),在代碼中硬編碼。聲明式事務(wù),在配置文件中配置(推薦使用)
聲明式事務(wù)又分為兩種:基于XML的聲明式事務(wù)基于注解的聲明式事務(wù)。我一般都是通過注解來進行的事務(wù)控制。也就是@Transactional
2. 先簡單介紹一下@Transactional注解嗎?項目中如何使用的?有哪些注意點嗎?
我們都是把注解加到需要使用事務(wù)控制的方法上,也可以加到類上,加到類上是給類里的所有的方法都加了事務(wù),不建議這樣做,這樣會增加不需要使用事務(wù)的接口的響應時長。
@Transactional注解只能用在public 方法上,如果用在protected或者private的方法上,不會報錯,但是該注解不會生效。
@Transactional注解只能回滾非檢查型異常,具體為RuntimeException及其子類。
3. Spring 事務(wù)中的隔離級別有哪幾種?
TransactionDefinition 接口中定義了五個表示隔離級別的常量:
TransactionDefinition.ISOLATION_DEFAULT: 使用后端數(shù)據(jù)庫默認的隔離級別,
Mysql 默認采用的 REPEATABLE_READ隔離級別 Oracle 默認采用的 READ_COMMITTED隔離級別。
TransactionDefinition.ISOLATION_READ_UNCOMMITTED: 最低的隔離級別,允許讀取未提交的數(shù)據(jù)變更,可能會導致臟讀、幻讀或不可重復讀。
TransactionDefinition.ISOLATION_READ_COMMITTED: 允許讀取并發(fā)事務(wù)已經(jīng)提交的數(shù)據(jù),可以阻止臟讀,但是幻讀或不可重復讀仍有可能發(fā)生。
TransactionDefinition.ISOLATION_REPEATABLE_READ: 對同一字段的多次讀取結(jié)果都是一致的,除非數(shù)據(jù)是被本身事務(wù)自己所修改,可以阻止臟讀和不可重復讀,但幻讀仍有可能發(fā)生。
TransactionDefinition.ISOLATION_SERIALIZABLE: 最高的隔離級別,完全服從ACID的隔離級別。所有的事務(wù)依次逐個執(zhí)行,這樣事務(wù)之間就完全不可能產(chǎn)生干擾,也就是說,該級別可以防止臟讀、不可重復讀以及幻讀。但是這將嚴重影響程序的性能。通常情況下也不會用到該級別。
4. Spring 事務(wù)中哪幾種事務(wù)傳播行為?
支持當前事務(wù)的情況:
TransactionDefinition.PROPAGATION_REQUIRED: 如果當前存在事務(wù),則加入該事務(wù);
如果當前沒有事務(wù),則創(chuàng)建一個新的事務(wù)。
TransactionDefinition.PROPAGATION_SUPPORTS: 如果當前存在事務(wù),則加入該事務(wù);
如果當前沒有事務(wù),則以非事務(wù)的方式繼續(xù)運行。
TransactionDefinition.PROPAGATION_MANDATORY: 如果當前存在事務(wù),則加入該事務(wù);如果當前沒有事務(wù),則拋出異常。(mandatory:強制性)不支持當前事務(wù)的情況:
不支持當前事務(wù)的情況:
TransactionDefinition.PROPAGATION_REQUIRES_NEW: 創(chuàng)建一個新的事務(wù),
如果當前存在事務(wù),則把當前事務(wù)掛起。
TransactionDefinition.PROPAGATION_NOT_SUPPORTED: 以非事務(wù)方式運行,
如果當前存在事務(wù),則把當前事務(wù)掛起。
TransactionDefinition.PROPAGATION_NEVER: 以非事務(wù)方式運行,
如果當前存在事務(wù),則拋出異常。
其他情況:
TransactionDefinition.PROPAGATION_NESTED: 如果當前存在事務(wù),則創(chuàng)建一個事務(wù)作為當前事務(wù)的嵌套事務(wù)來運行;如果當前沒有事務(wù),則該取值等價于TransactionDefinition.PROPAGATION_REQUIRED。
5. @Transactional注解只能用在public 方法上,這是為什么?
Spring事務(wù)的實現(xiàn)都是依靠AOP,本質(zhì)上也是依靠代理來實現(xiàn)。事務(wù)在spring中的實現(xiàn)其實就是生成bean對象的代理對象。
在bean進行創(chuàng)建出實例時, 如果是有事務(wù)注解的方法,就會被進行增強,最終形成代理類。在spring中,有兩種動態(tài)代理的方式,一種是jdk,它是將原始對象放入代理對象內(nèi)部,通過調(diào)用內(nèi)含的原始對象來實現(xiàn)原始的業(yè)務(wù)邏輯,而另一種是cglib,它是通過生成原始對象的子類,子類復寫父類的方法,從而實現(xiàn)對父類的增強。
jdk中,如果是private的方法,顯然是無法訪問的,而在cglib中,也是同樣。所以總結(jié)來說private方法不能被繼承,final方法不能被重寫,static方法和繼承不相干,所以它們3個的事務(wù)不起作用。
Spring選擇讓protected方法和package方法不支持事務(wù),所以只有public方法支持事務(wù)
6. 一個類中沒加事務(wù)的方法調(diào)用加事務(wù)的方法,為什么事務(wù)失效?怎么解決Spring事務(wù)失效的問題?
原因:
Spring事務(wù)管理用的是AOP,AOP底層用的是動態(tài)代理。所以spring 在掃描bean的時候會掃描方法上是否包含@Transactional注解,如果包含,spring會為這個bean動態(tài)地生成一個子類(即代理類,proxy),當這個有注解的方法被調(diào)用的時候,實際上是由代理類來調(diào)用的,代理類在調(diào)用之前就會啟動transaction。然而,如果這個有注解的方法是被同一個類中的其他方法調(diào)用的,那么該方法的調(diào)用并沒有通過代理類,而是直接通過原來的那個bean,所以就不會啟動transaction。
解決方式:
- 把方法B抽離到另外一個XXService中去,并且在這個Service中注入XXService,使用XXService調(diào)用方法B;
- 通過在方法內(nèi)部獲得當前類代理對象的方式,通過代理對象調(diào)用方法B
上面說了:動態(tài)代理最終都是要調(diào)用原始對象的,而原始對象在去調(diào)用方法時,是不會再觸發(fā)代理了!所以我們就使用代理對象來調(diào)用,就會觸發(fā)事務(wù);
綜上解決方案,那怎么獲取代理對象呢? 這里提供兩種方式:
- 使用 ApplicationContext 上下文對象獲取該對象;
- 使用 AopContext.currentProxy() 獲取代理對象,但是需要配置exposeProxy=true
7. 同一個類中標有@Transactional 的方法A,調(diào)用另一個標有@Transactional的 方法B會開啟幾個事務(wù)?
一個事務(wù)
Spring事務(wù)管理用的是AOP,AOP底層用的是動態(tài)代理。所以spring 在掃描bean的時候會掃描方法上是否包含@Transactional注解,如果包含,spring會為這個bean動態(tài)地生成一個子類(即代理類,proxy),代理類是繼承原來那個bean的。
此時,當這個有注解的方法被調(diào)用的時候,實際上是由代理類來調(diào)用的,代理類在調(diào)用之前就會啟動transaction。如果是被同一個類中的其他方法調(diào)用的,那么該方法的調(diào)用并沒有通過代理類,而是直接通過原來的那個bean,所以就不會啟動transaction,我們說只有一個事務(wù)。
8. 那么如何才能讓上面兩個方法開啟兩個事務(wù)呢?
1.把方法B抽離到另外一個XXService中去,在這個Service中注入XXService,使用XXService調(diào)用方法B;
2.通過在方法內(nèi)部獲得當前類代理對象的方式,通過代理對象調(diào)用方法B
上面說了:動態(tài)代理最終都是要調(diào)用原始對象的,而原始對象在去調(diào)用方法時,是不會再觸發(fā)代理了!所以我們就使用代理對象來調(diào)用,就會觸發(fā)事務(wù);
綜上解決方案,那怎么獲取代理對象呢? 這里提供兩種方式:
1.使用 ApplicationContext 上下文對象獲取該對象;
2.使用 AopContext.currentProxy() 獲取代理對象,但是需要配置exposeProxy=true
TestService testService = (TestService)AopContext.currentProxy();
testService.B();
同時還需要在B方法將傳播行為配置為 @Transactional(propagation = Propagation.REQUIRES_NEW)
9. @Transactional實現(xiàn)原理
注解介紹
@Transactional是spring中聲明式事務(wù)管理的注解配置方式。@Transactional注解可以幫助我們把事務(wù)開啟、提交或者回滾的操作,通過aop的方式進行管理。
通過@Transactional注解就能讓spring為我們管理事務(wù),免去了重復的事務(wù)管理邏輯,減少對業(yè)務(wù)代碼的侵入,使我們開發(fā)人員能夠?qū)W⒂跇I(yè)務(wù)層面開發(fā)。
我們知道實現(xiàn)@Transactional原理是基于spring aop,aop又是動態(tài)代理模式的實現(xiàn),下面我們就詳細分析一下實現(xiàn)原理
實現(xiàn)原理
猜想
- 首先,對于spring中aop實現(xiàn)原理有了解的話,應該知道想要對一個方法進行代理的話,肯定需要定義切點。在@Transactional的實現(xiàn)中,同樣如此,spring為我們定義了以 @Transactional 注解為植入點的切點,這樣才能知道@Transactional注解標注的方法需要被代理。
- 有了切面定義之后,在spring的bean的初始化過程中,就需要對實例化的bean進行代理,并且生成代理對象。
- 生成代理對象的代理邏輯中,進行方法調(diào)用時,需要先獲取切面邏輯,@Transactional注解的切面邏輯類似于@Around,在spring中是實現(xiàn)一種類似代理邏輯。
源碼分析
- 在配置好注解驅(qū)動方式的事務(wù)管理之后,spring會在ioc容器創(chuàng)建一個BeanFactoryTransactionAttributeSourceAdvisor實例,這個實例可以看作是一個切點,在判斷一個bean在初始化過程中是否需要創(chuàng)建代理對象,都需要驗證一次BeanFactoryTransactionAttributeSourceAdvisor是否是適用這個bean的切點。如果是,就需要創(chuàng)建代理對象,并且把BeanFactoryTransactionAttributeSourceAdvisor實例注入到代理對象中。
- 分析代理的對象發(fā)現(xiàn),最終的代理對象的代理方法是DynamicAdvisedInterceptor#intercept,分析這個方法后發(fā)現(xiàn)他最終調(diào)用的是TransactionInterceptor#invoke方法,并且把CglibMethodInvocation注入到invoke方法中,從上面可以看到CglibMethodInvocation是包裝了目標對象的方法調(diào)用的所有必須信息,因此,在TransactionInterceptor#invoke里面也是可以調(diào)用目標方法的,并且還可以實現(xiàn)類似@Around的邏輯,在目標方法調(diào)用前后繼續(xù)注入一些其他邏輯,比如事務(wù)管理邏輯。
- 當我們調(diào)進去TransactionInterceptor#invoke方法發(fā)現(xiàn)其中的核心方法是invokeWithinTransaction。