本篇內容主要講解“怎么理解Spring事物抽象”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學習“怎么理解Spring事物抽象”吧!
Spring事務抽象的關鍵是事務策略的概念。事務策略由TransactionManager
定義,特別是用于命令式事務管理的org.springframework.transaction.PlatformTransactionManager
接口和用于響應式事務管理的org.springframework.transaction.ReactiveTransactionManager
接口。以下清單顯示了PlatformTransactionManager
API的定義:
public interface PlatformTransactionManager extends TransactionManager { TransactionStatus getTransaction(TransactionDefinition definition) throws TransactionException; void commit(TransactionStatus status) throws TransactionException; void rollback(TransactionStatus status) throws TransactionException; }
盡管你可以從應用程序代碼中以編程方式使用它,但它主要是一個服務提供接口(SPI)。由于PlatformTransactionManager
是接口,因此可以根據需要輕松對其進行模擬或存根。它與JNDI
之類的查找策略無關。與Spring框架IoC容器中的任何其他對象(或bean)一樣,定義了PlatformTransactionManager
實現。這一優點使Spring框架事務成為值得抽象的,即使在使用JTA
時也是如此。與直接使用JTA
相比,你可以更輕松地測試事務代碼。
同樣,為了與Spring的理念保持一致,可以由任何PlatformTransactionManager
接口方法拋出的TransactionException
未檢查異常(也就是說,它擴展了java.lang.RuntimeException
類)。事物基礎架構故障幾乎總是致命的。在極少數情況下,應用程序代碼實際上可以從事務失敗中恢復,應用程序開發人員仍然可以選擇捕獲和處理TransactionException
。實際一點是,開發人員沒有被迫這樣做。
getTransaction(..)
方法根據TransactionDefinition
參數返回TransactionStatus
對象。如果當前調用堆棧中存在匹配的事務,則返回的TransactionStatus
可能表示一個新事務或一個現有事務。后一種情況的含義是,與Java EE事務上下文一樣,TransactionStatus
與執行線程相關聯。
從Spring框架5.2開始,Spring還為使用響應式類型或Kotlin
協程的響應式應用程序提供了事務管理抽象。以下清單顯示了由org.springframework.transaction.ReactiveTransactionManager
定義的事務策略:
public interface ReactiveTransactionManager extends TransactionManager { Mono<ReactiveTransaction> getReactiveTransaction(TransactionDefinition definition) throws TransactionException; Mono<Void> commit(ReactiveTransaction status) throws TransactionException; Mono<Void> rollback(ReactiveTransaction status) throws TransactionException; }
響應式事務管理器主要是服務提供接口(SPI),盡管你可以從應用程序代碼中以編程方式使用它。由于ReactiveTransactionManager
是接口,因此可以根據需要輕松對其進行模擬或存根。
TransactionDefinition
接口指定:
傳播:通常,事務范圍內的所有代碼都在該事務中運行。但是,如果在已存在事務上下文的情況下運行事務方法,則可以指定行為。例如,代碼可以在現有事務中繼續運行(常見情況),或者可以暫?,F有事務并創建新事務。Spring提供了EJB
CMT
熟悉的所有事務傳播選項。要了解有關Spring中事務傳播的語義的信息,請參閱事務傳播。
隔離:此事務與其他事務的工作隔離的程度。例如,此事務能否看到其他事務未提交的寫入?
超時:該事務在超時之前將運行多長時間,并由基礎事務基礎結構自動回滾。
只讀狀態:當代碼讀取但不修改數據時,可以使用只讀事務。在某些情況下,例如使用Hibernate
時,只讀事務可能是有用的優化。
這些設置反映了標準的事物概念。如有必要,請參考討論事務隔離級別和其他核心事務概念的資源。了解這些概念對于使用Spring框架或任何事務管理解決方案至關重要。
TransactionStatus
接口為事務代碼提供了一種控制事務執行和查詢事務狀態的簡單方法。這些概念應該很熟悉,因為它們對于所有事務API都是通用的。以下清單顯示了TransactionStatus
接口:
public interface TransactionStatus extends TransactionExecution, SavepointManager, Flushable { @Override boolean isNewTransaction(); boolean hasSavepoint(); @Override void setRollbackOnly(); @Override boolean isRollbackOnly(); void flush(); @Override boolean isCompleted(); }
無論你在Spring中選擇聲明式還是編程式事務管理,定義正確的TransactionManager
實現都是絕對必要的。通常,你可以通過依賴注入來定義此實現。TransactionManager
實現通常需要了解其工作環境:JDBC
、JTA
、Hibernate
等。
TransactionManager
實現通常需要了解其工作環境:JDBC
、JTA
、Hibernate
等。以下示例顯示了如何定義本地PlatformTransactionManager
實現(在這種情況下,使用純JDBC
)。
你可以通過創建類似于以下內容的bean來定義JDBC
數據源:
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close"> <property name="driverClassName" value="${jdbc.driverClassName}" /> <property name="url" value="${jdbc.url}" /> <property name="username" value="${jdbc.username}" /> <property name="password" value="${jdbc.password}" /> </bean>
然后,相關的PlatformTransactionManager
Bean定義將引用DataSource
定義。它應類似于以下示例:
<bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="dataSource"/> </bean>
如果你在Java EE容器中使用JTA
,則可以使用通過JNDI
獲得的容器DataSource
以及Spring的JtaTransactionManager
。以下示例顯示了JTA
和JNDI
查找:
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:jee="http://www.springframework.org/schema/jee" xsi:schemaLocation=" http://www.springframework.org/schema/beans https://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/jee https://www.springframework.org/schema/jee/spring-jee.xsd"> <jee:jndi-lookup id="dataSource" jndi-name="jdbc/jpetstore"/> <bean id="txManager" class="org.springframework.transaction.jta.JtaTransactionManager" /> <!-- other <bean/> definitions here --> </beans>
JtaTransactionManager
不需要了解數據源(或任何其他特定資源),因為它使用了容器的全局事務管理基礎結構。
dataSource
bean的先前定義使用jee名稱空間中的<jndi-lookup />標記。有關更多信息,參考JEE Schema。
你還可以輕松使用Hibernate
本地事務,如以下示例所示。在這種情況下,你需要定義一個Hibernate
LocalSessionFactoryBean
,你的應用程序代碼可使用該Hibernate
LocalSessionFactoryBean
獲取Hibernate
Session
實例。
DataSource
bean定義與先前顯示的本地JDBC
示例相似,因此在以下示例中未顯示。
如果通過
JNDI
查找數據源(由任何非JTA
事務管理器使用)并由Java EE容器管理,則該數據源應該是非事務性的,因為Spring框架(而不是Java EE容器)管理事務。
在這種情況下,txManager
bean是HibernateTransactionManager
類型。就像DataSourceTransactionManager
需要引用數據源一樣,HibernateTransactionManager
需要引用SessionFactory
。以下示例聲明了sessionFactory
和txManager
bean:
<bean id="sessionFactory" class="org.springframework.orm.hibernate5.LocalSessionFactoryBean"> <property name="dataSource" ref="dataSource"/> <property name="mappingResources"> <list> <value>org/springframework/samples/petclinic/hibernate/petclinic.hbm.xml</value> </list> </property> <property name="hibernateProperties"> <value> hibernate.dialect=${hibernate.dialect} </value> </property> </bean> <bean id="txManager" class="org.springframework.orm.hibernate5.HibernateTransactionManager"> <property name="sessionFactory" ref="sessionFactory"/> </bean>
如果使用Hibernate
和Java EE容器管理的JTA
事務,則應使用與前面的JDBC
JTA
示例相同的JtaTransactionManager
,如以下示例所示:
<bean id="txManager" class="org.springframework.transaction.jta.JtaTransactionManager"/>
如果使用
JTA
,則無論使用哪種數據訪問技術(無論是JDBC
、Hibernate
JPA
或任何其他受支持的技術),事務管理器定義都應該相同。這是由于JTA
事務是全局事務,它可以征用任何事務資源。
在所有這些情況下,無需更改應用程序代碼。你可以僅通過更改配置來更改事務的管理方式,即使更改意味著從本地事務轉移到全局事務,反之亦然。
到此,相信大家對“怎么理解Spring事物抽象”有了更深的了解,不妨來實際操作一番吧!這里是億速云網站,更多相關內容可以進入相關頻道進行查詢,關注我們,繼續學習!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。