Spring超詳細講解事務

什麼是事務

一個數據庫事務是一個被視為一個工作單元的邏輯上的一組操作,這些操作要麼全部執行,要麼全部不執行。

需要註意的是,並不是所有的數據庫(引擎)都支持事務,比如說MySQL的MyISAM存儲引擎

事務的四個特性(ACID)

原子性:事務是一個原子性操作,一個事務由一系列操作組成,這一系列操作要麼全部執行完成,要麼全部不執行

一致性:一個事務執行完成(不管成功還是失敗),數據庫的完整性不能被破壞(唯一約束,外鍵約束等)

隔離性:數據庫允許多個並發事務同時對數據進行讀寫操作,隔離性可以防止這些操作同時執行時導致的數據不一致的問題。事務分為不同的隔離級別:讀未提交(read uncommitted)、讀已提交(read committed)、可重復讀(repeatable read)和串行化(Serializable)

持久性:持久性意味著事務完成之後,該事物對數據的更改會永久的保存在數據庫中。

Spring對事務的支持

Spring支持編程式事務管理和聲明式事務管理兩種類型的事務管理。

編程式事務管理比較靈活,但是很難維護,平時用的很少。

聲明式事務管理易用性更高,侵入性、耦合度低,更推薦使用。

編程式事務管理

使用TransactionTemplate進行編程式事務管理

    @Autowired
    TransactionTemplate transactionTemplate;
    public void doTest() {
        transactionTemplate.execute(new TransactionCallbackWithoutResult() {
            @Override
            protected void doInTransactionWithoutResult(TransactionStatus transactionStatus) {
                try {
                    // 業務代碼
                    userMapper.deleteById(200106);
                    int i = 10/0;
                } catch (Exception e) {
                    transactionStatus.setRollbackOnly();
                }
            }
        });
    }

使用PlatformTransactionManager進行編程式事務管理

    @Autowired
    PlatformTransactionManager platformTransactionManager;
    public void doTest() {
        TransactionStatus transaction = platformTransactionManager.getTransaction(new DefaultTransactionDefinition());
        try{
            // 業務代碼
            userMapper.deleteById(200106);
            int i = 10/0;
            platformTransactionManager.commit(transaction);
        }catch (Exception e){
            platformTransactionManager.rollback(transaction);
        }
    }

聲明式事務管理

Spring主要通過兩種方式實現聲明式事務管理:

基於註解的聲明式事務管理

基於XML的聲明式事務管理

基於註解的聲明式事務管理

註解的事務管理主要靠@Transactional註解來實現的。

@Transactional註解可以作用於類或方法上,作用於類上時,該類的所有public方法都具有該事物屬性

@Transactional註解隻應該被應用到public修飾的方法上,這是有Spring AOP的本質決定的

    @Transactional
    public void doTest() {
        userMapper.deleteById(200106);
        int i = 10/0;
    }

@Transactional註解

@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface Transactional {
	// 指定事務管理器
	@AliasFor("transactionManager")
	String value() default "";
	@AliasFor("value")
	String transactionManager() default "";
	// 事務的傳播行為(enum)
	Propagation propagation() default Propagation.REQUIRED;
	// 事務的隔離級別(enum)
	Isolation isolation() default Isolation.DEFAULT;
	// 事務超時時間
	int timeout() default TransactionDefinition.TIMEOUT_DEFAULT;
	// 讀寫或隻讀事務,默認讀寫事務
	boolean readOnly() default false;
	// 導致事務回滾的異常類
	Class<? extends Throwable>[] rollbackFor() default {};
	// 導致事務回滾的異常類名
    // noRollbackForClassName = {"ArithmeticException"} 
	String[] rollbackForClassName() default {};
	// 不會導致事務回滾的異常類
	Class<? extends Throwable>[] noRollbackFor() default {};
	// 不會導致事務回滾的異常類名
	String[] noRollbackForClassName() default {};
}

Spring事務管理的三個接口

Spring對事務的操作主要通過PlatformTransactionManager接口,通過這個接口,Spring為各大平臺提供瞭對應的事務管理器,如 JDBC(org.springframework.jdbc.datasource.DataSourceTransactionManager )等。

在Spring框架中,事務管理相關的最重要的三個接口如下:

PlatformTransactionManager

public interface PlatformTransactionManager extends TransactionManager {
    // 根據指定的傳播行為,該方法返回當前活動事務或創建一個新的事務
	TransactionStatus getTransaction(@Nullable TransactionDefinition definition)
			throws TransactionException;
    // 提交指定的事務和關於它的狀態
	void commit(TransactionStatus status) throws TransactionException;
	// 回滾事務
	void rollback(TransactionStatus status) throws TransactionException;
}

TransactionDefinition

public interface TransactionDefinition {
    // …………
    // 返回傳播行為
	default int getPropagationBehavior() {
		return PROPAGATION_REQUIRED;
	}
    // 返回該事務隔離級別
	default int getIsolationLevel() {
		return ISOLATION_DEFAULT;
	}
    // 返回超時時間
	default int getTimeout() {
		return TIMEOUT_DEFAULT;
	}
    // 返回是否隻讀
	default boolean isReadOnly() {
		return false;
	}
    // 返回事務名稱
	@Nullable
	default String getName() {
		return null;
	}
	static TransactionDefinition withDefaults() {
		return StaticTransactionDefinition.INSTANCE;
	}
}

TransactionStatus

public interface TransactionStatus extends TransactionExecution, SavepointManager, Flushable {
	boolean hasSavepoint();
	@Override
	void flush();
}

Spring事務屬性

Spring事務管理的五大屬性:隔離級別、傳播行為、超時時間、是否隻讀、回滾規則

TransactionDefinition接口中定義瞭事務的隔離級別、傳播行為、超時時間的常量

public interface TransactionDefinition {
    // 當前存在事務,則加入該事務;如果當前沒有事務,則創建一個新的事務
	int PROPAGATION_REQUIRED = 0;
    // ~,~,~,則繼續以非事務的方式運行
	int PROPAGATION_SUPPORTS = 1;
    // ~,~,~,則拋出異常
	int PROPAGATION_MANDATORY = 2;
    // 創建一個新事務,如果當前存在事務,則把當前事務掛起
	int PROPAGATION_REQUIRES_NEW = 3;
    // 以非事務的方式運行,如果當前存在事務,則把當前事務掛起
	int PROPAGATION_NOT_SUPPORTED = 4;
    // 以非事務的方式運行,如果當前存在事務,則拋出異常
	int PROPAGATION_NEVER = 5;
    // ~,則創建一個事務作為當前事務的嵌套事務來運行,~,則等價於PROPAGATION_REQUIRED
	int PROPAGATION_NESTED = 6;
    // 默認的隔離級別
	int ISOLATION_DEFAULT = -1;
    // 讀未提交
	int ISOLATION_READ_UNCOMMITTED = 1; 
    // 讀已提交
	int ISOLATION_READ_COMMITTED = 2;  
    // 可重復讀
	int ISOLATION_REPEATABLE_READ = 4; 
    // 序列化讀
	int ISOLATION_SERIALIZABLE = 8; 
    // 默認的超時時間,單位:秒
    // 所謂事務超時,就是規定一個事務允許執行的最長時間,如果超過瞭該時間事務還沒有執行完成,則回滾事務。
	int TIMEOUT_DEFAULT = -1;
	default int getPropagationBehavior() {
		return PROPAGATION_REQUIRED;
	}
	default int getIsolationLevel() {
		return ISOLATION_DEFAULT;
	}
	default int getTimeout() {
		return TIMEOUT_DEFAULT;
	}
	default boolean isReadOnly() {
		return false;
	}
	@Nullable
	default String getName() {
		return null;
	}
	static TransactionDefinition withDefaults() {
		return StaticTransactionDefinition.INSTANCE;
	}
}

是否隻讀:指該事物是隻讀操作還是讀寫操作,boolean類型

回滾規則:我們可以人為的控制事務在執行過程中拋出某些異常時仍然提交事務,或者拋出某些異常時回滾事務

到此這篇關於Spring超詳細講解事務的文章就介紹到這瞭,更多相關Spring事務內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: