日日操夜夜添-日日操影院-日日草夜夜操-日日干干-精品一区二区三区波多野结衣-精品一区二区三区高清免费不卡

公告:魔扣目錄網為廣大站長提供免費收錄網站服務,提交前請做好本站友鏈:【 網站目錄:http://www.ylptlb.cn 】, 免友鏈快審服務(50元/站),

點擊這里在線咨詢客服
新站提交
  • 網站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會員:747

本文總結了Spring 聲明式事務的源碼實現、五種常見的事務失效情況,并提供了相應的解決方案。

一、前言

在Web 開發中,Spring 框架已經成為了眾多開發者的首選。Spring 的聲明式事務管理是其中最重要的特性之一,它可以幫助我們簡化業務邏輯的復雜度,并且確保在出現異常情況時數據的一致性。

事務失效情況很常見,但我們只要注意,就可以避免事情發生!在本文中,我將詳細地介紹 Spring 聲明式事務的源碼實現和事務失效常見的五種情況,并給出有效的解決方案。

其實我們常說的事務失效是聲明式事務(@Transactional)的失效,本文也是從聲明式事務來進行演示的!

通過本文的學習,你將掌握如何正確地使用 Spring 的事務管理,減少生產事故。

「一定要保持數據一致性」。

二、@Transactional注解參數解讀

我們拿出幾個經常使用的參數來簡單介紹一下:

  • propagation:指定事務的傳播行為。其取值包括 REQUIRED、SUPPORTS、MANDATORY、REQUIRES_NEW、NOT_SUPPORTED、NEVER 和 NESTED 等。默認為 REQUIRED。 其中,REQUIRED 表示如果當前已經存在一個事務,則加入該事務,否則新建一個事務;而 REQUIRES_NEW 表示新建一個獨立的事務,如果當前已經存在事務,則掛起當前事務。后面就不一一說了,大家可以自行百度哈!
  • isolation:指定事務的隔離級別。其取值包括 DEFAULT、READ_UNCOMMITTED、READ_COMMITTED、REPEATABLE_READ 和 SERIALIZABLE 等。默認為 DEFAULT。 其中,DEFAULT 表示采用數據庫的默認隔離級別.
  • timeout:指定事務的超時時間,單位為秒。默認為 -1,表示不設置超時時間。如果在規定時間內事務還未完成,則拋出 TransactionTimedOutException 異常。
  • readOnly:指定事務是否只讀,即是否允許修改數據。默認為 false,表示可以進行數據修改操作。如果將其設置為 true,則表示該事務僅能進行數據查詢操作,不能進行數據修改操作,這樣可以提高并發性能。
  • rollbackFor:指定哪些異常需要回滾事務。其取值為一個 Class 數組,其中每個元素表示一個異常類型。默認為空,表示只有拋出 RuntimeException 或 Error 類型的異常時才回滾事務。
  • noRollbackFor:指定哪些異常不需要回滾事務。其取值為一個 Class 數組,其中每個元素表示一個異常類型。默認為空,表示拋出任何異常都回滾事務。

三、聲明式事務源碼實現

聲明式事務實現類為:TransactionInterceptor ,下面我們來一起看看這個類!

源碼版本為Springboot2.7.1。

public class TransactionInterceptor extends TransactionAspectSupport 
 implements MethodInterceptor, Serializable{}

TransactionInterceptor UML圖:

圖片

聲明式事務主要是通過AOP實現,主要包括以下幾個節點:

  1. 啟動時掃描@Transactional注解:在啟動時,Spring Boot會掃描所有使用了@Transactional注解的方法,并將其封裝成TransactionAnnotationParser對象。
  2. AOP 來實現事務管理的核心類依然是 TransactionInterceptor。TransactionInterceptor 是一個攔截器,用于攔截使用了 @Transactional 注解的方法
  3. 將TransactionInterceptor織入到目標方法中:在AOP編程中,使用AspectJ編寫切面類,通過@Around注解將TransactionInterceptor織入到目標方法中。
  4. 在目標方法執行前創建事務:在目標方法執行前,TransactionInterceptor會調用PlatformTransactionManager創建一個新的事務,并將其納入到當前線程的事務上下文中。
  5. 執行目標方法:在目標方法執行時,如果發生異常,則將事務狀態標記為ROLLBACK_ONLY;否則,將事務狀態標記為COMMIT。
  6. 提交或回滾事務:在目標方法執行完成后,TransactionInterceptor會根據事務狀態(COMMIT或ROLLBACK_ONLY)來決定是否提交或回滾事務。

源碼:

@Override
@Nullable
public Object invoke(MethodInvocation invocation) throws Throwable {
 // Work out the target class: may be {@code null}.
 // The TransactionAttributeSource should be passed the target class
 // as well as the method, which may be from an interface.
 Class<?> targetClass = (invocation.getThis() != null ? AopUtils.getTargetClass(invocation.getThis()) : null);

 // Adapt to TransactionAspectSupport's invokeWithinTransaction...
 return invokeWithinTransaction(invocation.getMethod(), targetClass, new CoroutinesInvocationCallback() {
  @Override
  @Nullable
  public Object proceedWithInvocation() throws Throwable {
   return invocation.proceed();
  }
  @Override
  public Object getTarget() {
   return invocation.getThis();
  }
  @Override
  public Object[] getArguments() {
   return invocation.getArguments();
  }
 });
}

下面是核心處理方法,把不太重要的代碼忽略了,留下每一步的節點。

@Nullable
protected Object invokeWithinTransaction(Method method, @Nullable Class<?> targetClass,
 final InvocationCallback invocation) throws Throwable {
 // 獲取事務屬性
 final TransactionManager tm = determ.NETransactionManager(txAttr);
 // 準備事務
 TransactionInfo txInfo = prepareTransactionInfo(ptm, txAttr, joinpointIdentification, status);
 // 執行目標方法
 Object retVal = invocation.proceedWithInvocation();
  // 回滾事務
 completeTransactionAfterThrowing(txInfo, ex);
 // 提交事務
 commitTransactionAfterReturning(txInfo);
}

四、五種失效和解決方案

圖片

下面我們從幾個情況來給大家展示失效場景并給出解決方案。

1、類沒有被 Spring 管理

public class UserServiceImpl implements UserService {

    @Autowired
    private UserDao userDao;

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void addUser(User user) {
        userDao.addUser(user);
    }
}

如上代碼所示,UserServiceImpl 類沒有被聲明為 Spring Bean,因此其中的 addUser() 方法無法受到 Spring 事務管理的保護。 我們使用Spring,要把類交給Spring進行管理,不然是無法生效!

「解決方案:」 交給spring進行管理bean,在類上添加:@Service!

2、方法不是public修飾

@Service
public class UserService {

    @Autowired
    private UserDao userDao;

    @Transactional(rollbackFor = Exception.class)
    protected void addUser(User user) {
        userDao.addUser(user);
    }
}

我們上面說了聲明式事務是基于AOP實現的,AOP是通過代理模式實現的,即為目標對象生成一個代理對象,當調用代理對象的方法時,會自動添加事務的控制代碼。 在這種情況下,如果事務注釋所在的方法不是public的,則無法生成代理對象,因此事務代碼將無法添加到方法執行前后,導致事務失效。

其實這種情況還是不經常這么使用,我們基本都是使用接口和實現大部分都是public修飾的!

「解決方案:」 使用public來修飾方法。

3、異常被捕獲并處理了

@Service
public class UserServiceImpl implements UserService {

    @Autowired
    private UserDao userDao;

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void addUser(User user) {
        try {
            userDao.addUser(user);
        } catch (Exception e) {
            // 處理異常,但沒有拋出或重新拋出異常
            log.error("add user error", e);
        }
    }
}

如上代碼所示,如果 userDao.addUser() 方法拋出異常,但是在 UserServiceImpl.addUser() 中被捕獲并處理了,事務檢測不到有異常拋出,那么事務不會回滾。

「解決方案:」 catch 處理完成后,在重新把異常在拋出去:throw e。

4、同一個類中,方法內部調用

@Service
public class UserServiceImpl implements UserService {

    @Autowired
    private UserDao userDao;

    @Override
    public void addUser(User user) {
        doAddUser(user);
    }

    @Transactional(rollbackFor = Exception.class)
    public void doAddUser(User user) {
        userDao.addUser(user);
    }
}

Spring使用代理來實現事務控制,但是這種方法直接調用了this對象的方法,則無法通過代理來攔截該方法調用,從而使得事務失效。

「解決方案:」

推薦使用有兩種:

  • 使用ApplicationContext來獲取當前bean對象來調用doAddUser方法。
  • 在addUser方法加上@Transactional(rollbackFor = Exception.class)。

網上還有一些使用AopContext.currentProxy()拿到代理對象的、自己注入自己的、抽到單獨的bean里的 這里小編不是很推薦!

方法一完整展示:

如果覺得Service里注入ApplicationContext 不優雅,可以抽到單獨的工具bean里!

@Service
public class UserServiceImpl implements UserService {

 @Autowired
    private UserDao userDao;
    @Autowired
 private ApplicationContext applicationContext;

    @Override
    public void addUser(User user) {
     UserServiceImpl userService = applicationContext.getBean(UserServiceImpl.class);
        userService.doAddUser(user);
    }

    @Transactional(rollbackFor = Exception.class)
    public void doAddUser(User user) {
        userDao.addUser(user);
    }
}

5、MySQL存儲引警不支持事務

MyISAM 存儲引擎是 MySQL 的一種存儲引擎,它是 MySQL 5.1 版本之前的默認存儲引擎,它是不支持事務的。從 MySQL 5.5 版本開始,InnoDB 成為了 MySQL 的默認存儲引擎。我們想使用也可以切換到MyISAM引擎。

「解決方案:」 把mysql換到5.5以上使用InnoDB 存儲引擎。

「補充使用MyISAM 方式:」

  • 表從 InnoDB 引擎轉換為 MyISAM 引擎:使用 ALTER TABLE 命令來更改表的引擎類型。
ALTER TABLE table_name ENGINE = MyISAM;
  • 默認的存儲引擎設置為 MyISAM, 可以在 MySQL 配置文件中設置 default-storage-engine 參數。
 
default-storage-engine=MyISAM
  • 創建表時指定MyISAM 引擎 要將表的引擎類型設置為 MyISAM,請在 CREATE TABLE 語句中包含 ENGINE = MyISAM 子句
 
CREATE TABLE table_name (
    column1 datatype,
    column2 datatype,
    ...
) ENGINE = MyISAM;

五、總結

本文總結了Spring 聲明式事務的源碼實現、五種常見的事務失效情況,并提供了相應的解決方案。

當然還有很多情況:被final修飾、多線程調用、傳播行為使用不當、拋的異常不對應等等

理解 Spring 事務機制的,深入了解 Spring 事務的內部原理。同時,在使用聲明式事務的過程中,我們也可以針對自己的業務場景進行定制化的配置,比如指定特定的事務傳播機制、設置超時時間等,這些都有助于更好地應對復雜的業務場景和代碼需求。這樣才能真正地提高系統的可維護性、可擴展性和穩定性。

分享到:
標簽:Spring
用戶無頭像

網友整理

注冊時間:

網站:5 個   小程序:0 個  文章:12 篇

  • 51998

    網站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會員

趕快注冊賬號,推廣您的網站吧!
最新入駐小程序

數獨大挑戰2018-06-03

數獨一種數學游戲,玩家需要根據9

答題星2018-06-03

您可以通過答題星輕松地創建試卷

全階人生考試2018-06-03

各種考試題,題庫,初中,高中,大學四六

運動步數有氧達人2018-06-03

記錄運動步數,積累氧氣值。還可偷

每日養生app2018-06-03

每日養生,天天健康

體育訓練成績評定2018-06-03

通用課目體育訓練成績評定