SpringBoot的事務管理你會了麼?

Springboot內部提供的事務管理器是根據autoconfigure來進行決定的。

比如當使用jpa的時候,也就是pom中加入了spring-boot-starter-data-jpa這個starter之後(之前我們分析過springboot的自動化配置原理)。

Springboot會構造一個JpaTransactionManager這個事務管理器。

而當我們使用spring-boot-starter-jdbc的時候,構造的事務管理器則是DataSourceTransactionManager。

這2個事務管理器都實現了spring中提供的PlatformTransactionManager接口,這個接口是spring的事務核心接口。

這個核心接口有以下這幾個常用的實現策略:

HibernateTransactionManager

DataSourceTransactionManager

JtaTransactionManager

JpaTransactionManager

具體的PlatformTransactionManager繼承關係如下:

spring-boot-starter-data-jpa這個starter會觸發HibernateJpaAutoConfiguration這個自動化配置類,HibernateJpaAutoConfiguration繼承了JpaBaseConfiguration基礎類。

在JpaBaseConfiguration中構造了事務管理器:

@Bean

@ConditionalOnMissingBean(PlatformTransactionManager.class)

public PlatformTransactionManager transactionManager() {

return new JpaTransactionManager();

}

spring-boot-starter-jdbc會觸發DataSourceTransactionManagerAutoConfiguration這個自動化配置類,也會構造事務管理器:

@Bean

@ConditionalOnMissingBean(PlatformTransactionManager.class)

@ConditionalOnBean(DataSource.class)

public DataSourceTransactionManager transactionManager() {

return new DataSourceTransactionManager(this.dataSource);

}

Spring的事務管理器PlatformTransactionManager接口中定義了3個方法:

// 基於事務的傳播特性,返回一個已經存在的事務或者創建一個新的事務

TransactionStatus getTransaction(TransactionDefinition definition) throwsTransactionException;

// 提交事務

void commit(TransactionStatus status) throws TransactionException;

// 回滾事務

void rollback(TransactionStatus status) throws TransactionException;

其中TransactionDefinition接口表示跟spring兼容的事務屬性,比如傳播行爲、隔離級別、超時時間、是否只讀等屬性。

DefaultTransactionDefinition類是一個默認的TransactionDefinition實現,它的傳播行爲是PROPAGATION_REQUIRED(如果當前沒事務,則創建一個,否則加入到當前事務中),隔離級別是數據庫默認級別。

TransactionStatus接口表示事務的狀態,比如事務是否是一個剛構造的事務、事務是否已經完成等狀態。

下面這段代碼就是傳統事務的常見寫法:

transaction.begin();

try {

...

transaction.commit();

catch(Exception e) {

...

transaction.rollback();

finally {

}

由於spring的事務操作被封裝到了PlatformTransactionManager接口中,commit和rollback方法對應接口中的方法,begin方法在getTransaction方法中會被調用。

細心的讀者發現文章前面構造事務管理器的時候都會加上這段註解:

@ConditionalOnMissingBean(PlatformTransactionManager.class)

也就是說如果我們手動配置了事務管理器,Springboot就不會再爲我們自動配置事務管理器。

如果要使用多個事務管理器的話,那麼需要手動配置多個:

@Configuration

public class DatabaseConfiguration {

@Bean

public PlatformTransactionManager transactionManager1(EntityManagerFactory entityManagerFactory) {

return new JpaTransactionManager(entityManagerFactory);

}

@Bean

public PlatformTransactionManager transactionManager2(DataSource dataSource) {

return new DataSourceTransactionManager(dataSource);

}

}

然後使用Transactional註解的時候需要聲明是哪個事務管理器:

@Transactional(value="transactionManager1")

public void save() {

doSave();

}

Spring給我們提供了一個TransactionManagementConfigurer接口,該接口只有一個方法返回PlatformTransactionManager。其中返回的PlatformTransactionManager就表示這是默認的事務處理器,這樣在Transactional註解上就不需要聲明是使用哪個事務管理器了。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章