精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

詳解Spring多線程下如何保證事務的一致性

開發 前端
我們先來大概的了解下Spring事務的工作原理,核心技術是通過AOP實現,將獲取的Connection對象綁定到當前線程上下文中(ThreadLocal)。

環境:Spring5.3.23

1. 事務原理

首先,我們先來大概的了解下Spring事務的工作原理,核心技術是通過AOP實現,將獲取的Connection對象綁定到當前線程上下文中(ThreadLocal)。

事務核心攔截器TransactionInterceptor對象,如下(以下只會列出核心代碼):

public class TransactionInterceptor {
  public Object invoke(MethodInvocation invocation) {
    // 該方法調用為核心方法,該方法在父類中
    return invokeWithinTransaction(...) ;
  }
}

父類TransactionAspectSupport

public abstract class TransactionAspectSupport {
  protected Object invokeWithinTransaction(...) {
    // 1.1.創建事務對象
    TransactionInfo txInfo = createTransactionIfNecessary(ptm, txAttr, joinpointIdentification);
    try {
        // 調用下一個攔截器或者是目標方法
      retVal = invocation.proceedWithInvocation();
    }
    catch (Throwable ex) {
      // 1.2.回滾事務
      completeTransactionAfterThrowing(txInfo, ex);
      throw ex;
    } finally {
      // 重置ThreadLocal中的TransactionInfo對象
      cleanupTransactionInfo(txInfo);
    }
    // 1.3.提交或者回滾事務
    commitTransactionAfterReturning(txInfo);
    return retVal;
  }  
}

上面代碼列出了主要的事務執行流程及動作,我們主要是關心數據庫連接對象Connection在當前線程中是如何使用的。

創建事務對象

protected TransactionInfo createTransactionIfNecessary(
    @Nullable PlatformTransactionManager tm,
    @Nullable TransactionAttribute txAttr, 
    final String joinpointIdentification) {
  TransactionStatus status = null;
  if (txAttr != null) {
    if (tm != null) {
      // 創建事務狀態對象
      status = tm.getTransaction(txAttr);
    }
  }
  // 將事務狀態對象包裝到TransactionInfo中,然后將這個對象綁定到當前線程中
  return prepareTransactionInfo(tm, txAttr, joinpointIdentification, status);
}

創建事務狀態對象

public abstract class AbstractPlatformTransactionManager {
  public final TransactionStatus getTransaction(...) {
    if (isExistingTransaction(transaction)) {
      // Existing transaction found -> check propagation behavior to find out how to behave.
      return handleExistingTransaction(def, transaction, debugEnabled);
    }


    // 如果超時時間 < -1則拋出異常
    if (def.getTimeout() < TransactionDefinition.TIMEOUT_DEFAULT) {
      throw new InvalidTimeoutException("Invalid transaction timeout", def.getTimeout());
    }


    // 當前不存在事務,則拋出異常
    if (def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_MANDATORY) {
      throw new IllegalTransactionStateException(
          "No existing transaction found for transaction marked with propagation 'mandatory'");
    }
    // 其它的傳播特性,開啟事務功能
    else if (def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRED ||
        def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRES_NEW ||
        def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_NESTED) {
      try {
        // 開始事務
        return startTransaction(def, transaction, debugEnabled, suspendedResources);
      }
    }
  }
}

開始事務

private TransactionStatus startTransaction(
    TransactionDefinition definition, 
    Object transaction,
    boolean debugEnabled, 
    @Nullable SuspendedResourcesHolder suspendedResources) {


  boolean newSynchronization = (getTransactionSynchronization() != SYNCHRONIZATION_NEVER);
  DefaultTransactionStatus status = newTransactionStatus(
      definition, transaction, true, newSynchronization, debugEnabled, suspendedResources);
  //     
  doBegin(transaction, definition);
  prepareSynchronization(status, definition);
  return status;
}

創建Connection對象,并綁定到當前線程

public class DataSourceTransactionManager {
  protected void doBegin(
      Object transaction, 
      TransactionDefinition definition) {
    DataSourceTransactionObject txObject = (DataSourceTransactionObject) transaction;
    Connection con = null;
    try {
      if (!txObject.hasConnectionHolder() ||
          txObject.getConnectionHolder().isSynchronizedWithTransaction()) {
        // 獲取數據庫連接對象  
        Connection newCon = obtainDataSource().getConnection();
        txObject.setConnectionHolder(new ConnectionHolder(newCon), true);
      }
      // 將連接對象綁定到當前的線程
      if (txObject.isNewConnectionHolder()) {
        TransactionSynchronizationManager.bindResource(obtainDataSource(), txObject.getConnectionHolder());
      }
    }
  }
}

到此,已經清楚了當開始一個新的事務時,Spring會將獲取的Connection綁定到當前的Thread中。

當我們使用通過JdbcTemplate操作數據庫時,如下:

public class JdbcTemplate {
  // 核心執行方法
  private <T> T execute(...) {
    // 獲取數據庫連接對象
    Connection con = DataSourceUtils.getConnection(obtainDataSource());
  }
}

DataSourceUtils

public abstract class DataSourceUtils {
  public static Connection getConnection(DataSource dataSource) throws CannotGetJdbcConnectionException {
    try {
      return doGetConnection(dataSource) ;
    }
  }
  public static Connection doGetConnection(DataSource dataSource) throws SQLException {
    // 通過TransactionSynchronizationManager從當前線程上下文中獲取連接對象
    // 在上面我們也是通過這個對象將連接對象綁定到當前的Thread中
    ConnectionHolder conHolder = (ConnectionHolder) TransactionSynchronizationManager.getResource(dataSource);
    if (conHolder != null && (conHolder.hasConnection() || conHolder.isSynchronizedWithTransaction())) {
      conHolder.requested() ;
      if (!conHolder.hasConnection()) {
        conHolder.setConnection(fetchConnection(dataSource)) ;
      }
      return conHolder.getConnection() ;
    }
  }
}

原理相信你應該非常清楚了,每個線程都會綁定自己的Connection。那在多線程下每個線程都使用的是自己的Connection對象,所以要想保證事務的一致性,單靠傳統的方式一個@Transaction是肯定無法解決的,接下來我們就來實現一個多線程下的事務一致性的處理。

2.多線程事務

多線程下要實現事務的一致性,我們需要借助JUC下的相關類來實現。

這里直接給出代碼示例:

static class PersonService {
  @Resource
  private JdbcTemplate jdbcTemplate;
  @Resource
  private DataSource dataSource ; 


  @Transactional
  public void save() throws Exception {
    CountDownLatch cdl = new CountDownLatch(2) ;
    AtomicBoolean txRollback = new AtomicBoolean(false) ;
    CompletableFuture.runAsync(() -> {
      Person person = new Person();
      person.setAge(1);
      person.setName("張三");
      transactionTemplate.execute(status -> {
        int result = 0 ;
        try {
          result = jdbcTemplate.update("insert into t_person (age, name) values (?, ?)", person.getAge(), person.getName()) ;
          // TODO
          // System.out.println(1 / 0) ;
        } catch (Exception e) {
            // 當發生異常后將狀態該為true
          txRollback.set(true) ;
        }
        try {
            // 計數減一
          cdl.countDown() ;
          // 繼續等待其它線程結束
          cdl.await() ;
        } catch (InterruptedException e) {
          e.printStackTrace();
        }
          // 如果回滾狀態為true說明有線程發生了異常,需要事務回滾
        if (txRollback.get()) {
          // 標記當前事務回滾
          status.setRollbackOnly() ;
        }
        System.out.printf("%s Insert Operator Result: %d 次%n", Thread.currentThread().getName(), result);
        return result ;
      }) ;
      }) ;
    transactionTemplate.execute(status -> {
      Person person = new Person();
      person.setAge(2);
      person.setName("李四");
      int result = 0 ;
      try {
        result = jdbcTemplate.update("insert into t_person (age, name) values (?, ?)", person.getAge(), person.getName()) ;
        // TODO
        TimeUnit.SECONDS.sleep(3) ;
      } catch (Exception e) {
        txRollback.set(true) ; 
      }
      try {
        cdl.countDown() ;
        cdl.await() ;
      } catch (InterruptedException e) {
        e.printStackTrace();
      }
      if (txRollback.get()) {
        // 回滾
        status.setRollbackOnly() ;
      }
      System.out.printf("%s Insert Operator Result: %d 次%n", Thread.currentThread().getName(), result);
      return result ;
    }) ;
    cdl.await() ;
    System.err.println("Operator Complete...") ;
  }
}

以上就是借助JUC來實現多線程下的事務一致性問題。

其實如果你真的理解了事務的原理,其實這里還有更加簡單的實現方式,大家可以先思考,咱們下期再說這種簡單的實現方法。

完畢!!!

責任編輯:武曉燕 來源: Spring全家桶實戰案例源碼
相關推薦

2021-03-04 06:49:53

RocketMQ事務

2022-10-19 12:22:53

并發扣款一致性

2020-01-02 09:06:23

微服務數據框架

2025-02-10 03:00:00

2022-08-29 08:38:00

事務一致性

2019-08-30 12:46:10

并發扣款查詢SQL

2020-08-05 08:46:10

NFS網絡文件系統

2025-03-27 08:20:54

2024-01-10 08:01:55

高并發場景悲觀鎖

2024-12-26 15:01:29

2023-09-07 08:11:24

Redis管道機制

2020-04-01 15:50:17

TiDBMySQL數據庫

2020-06-01 22:09:48

緩存緩存同步緩存誤用

2021-07-21 15:50:42

Serverless 業務部署

2021-12-14 07:15:57

MySQLRedis數據

2024-08-20 16:13:52

2023-05-26 07:34:50

RedisMySQL緩存

2024-10-28 12:41:25

2024-01-15 10:38:20

多級緩存數據一致性分布式緩存

2022-03-29 10:39:10

緩存數據庫數據
點贊
收藏

51CTO技術棧公眾號

99久久综合色| 欧美一区在线看| 欧美在线free| 一区二区三区四区视频在线观看 | 欧美午夜宅男影院在线观看| 国产日韩欧美亚洲一区| 国产成人一级片| 日韩欧美电影| 亚洲第一页自拍| 男女啪啪网站视频| 成全电影大全在线观看| 久久久国产精华| 99re在线播放| 国产91av在线播放| 亚洲国产1区| 色婷婷综合成人| 美女又爽又黄视频毛茸茸| 国产精品诱惑| 亚洲成av人影院在线观看网| 亚洲精品视频一二三| 色呦呦视频在线| 久久99精品国产麻豆婷婷| 97精品国产97久久久久久春色| 在线视频第一页| 女同另类激情重口| 91精品国产色综合久久不卡蜜臀 | 日韩黄色高清视频| 天天色天天干天天色| 写真福利精品福利在线观看| 亚洲最色的网站| 亚洲在线播放电影| 青青免费在线视频| 成人18精品视频| 91久久国产自产拍夜夜嗨| 久久这里只有精品9| 国产日本精品| 欧美极品少妇xxxxx| 小泽玛利亚一区| 操欧美老女人| 亚洲欧洲第一视频| 久久精品综合视频| 精品久久ai| 精品国产自在久精品国产| 免费av不卡在线| 国产精品天堂蜜av在线播放| 色婷婷久久99综合精品jk白丝| 日本一本中文字幕| 欧美aaaxxxx做受视频| 亚洲欧美激情小说另类| 亚洲欧美日韩精品久久久| 户外极限露出调教在线视频| 91麻豆国产福利在线观看| 国外成人在线视频网站| 亚洲第一色视频| 国产91精品露脸国语对白| 99国精产品一二二线| 性生交生活影碟片| 粉嫩av一区二区三区在线播放| 亚洲综合成人婷婷小说| 国产特黄一级片| 国产a视频精品免费观看| av一区和二区| 日本高清视频免费看| 波多野结衣中文字幕一区二区三区| 国产精品一区二区三区免费| 亚洲免费成人网| 波多野结衣在线一区| 国产亚洲自拍偷拍| 日韩av免费观影| 久久精品无码一区二区三区| 日韩一本精品| 国产黄色小视频在线| 亚洲欧美电影一区二区| 黄色小视频大全| av老司机在线观看| 在线视频欧美精品| 加勒比av中文字幕| 亚洲一区二区三区日本久久九| 精品国产区一区| 美国黄色a级片| 欧美好骚综合网| 久久99国产综合精品女同| 日韩精品久久久久久久酒店| 嫩草成人www欧美| 国产欧美精品xxxx另类| 国产富婆一级全黄大片| www.视频一区| 午夜精品短视频| 伊人222成人综合网| 欧美日韩国产精品一区二区不卡中文| 天天摸天天碰天天添| 亚洲日本免费电影| 亚洲精品动漫久久久久| 长河落日免费高清观看| 一区视频在线| 国产精品入口尤物| 懂色av蜜臀av粉嫩av分享吧| 久久久久成人黄色影片| 精品国产三级a∨在线| 成人小电影网站| 欧美一级久久久久久久大片| 野外性满足hd| 欧美激情视频一区二区三区免费| 欧美一级bbbbb性bbbb喷潮片| 亚洲视频一区二区三区四区| 成人美女视频在线看| 亚洲精品中文字幕乱码三区不卡| 黄污视频在线观看| 欧美日韩国产高清一区二区| 中文字幕人妻一区二区三区| 999精品在线| 欧美资源在线观看| 性猛交富婆╳xxx乱大交天津| 国产午夜亚洲精品不卡| 国产aaa免费视频| 国产精品1区| 一区二区三区视频观看| 亚洲精品午夜久久久久久久| 精品一区二区三区影院在线午夜| 久久人人97超碰人人澡爱香蕉| a毛片在线播放| 欧美色男人天堂| 国产传媒第一页| 在线 亚洲欧美在线综合一区| 国产精品免费一区二区三区都可以| 欧美一区二区黄片| 一区二区免费在线播放| 久久久久久久高清| 日韩精品一区二区久久| 国产成人一区二区三区小说| 日本精品一二区| 亚洲综合一区二区三区| 中文字幕丰满乱码| 欧美超碰在线| 成人激情电影一区二区| av资源网站在线观看| 色久综合一二码| 精品无码人妻一区| 性高湖久久久久久久久| 精品国产综合区久久久久久| 色yeye免费人成网站在线观看| 5858s免费视频成人| 国精品人伦一区二区三区蜜桃| 日本大胆欧美人术艺术动态| 欧美日韩一区二区视频在线| 深夜av在线| 日韩精品福利网站| 欧美啪啪小视频| 91免费看视频| 91蝌蚪视频在线观看| 久草成人在线| 国产精品视频午夜| 午夜伦理在线| 欧美人动与zoxxxx乱| 女同久久另类69精品国产 | 性折磨bdsm欧美激情另类| 亚洲精品网址| 91国产丝袜在线放| 69av成人| 亚洲精品自拍第一页| 久久亚洲精品国产| 国产性色一区二区| 亚洲欧美自偷自拍另类| 五月久久久综合一区二区小说| 国产美女直播视频一区| 国产剧情在线| 亚洲精品一线二线三线| 中文字字幕在线中文| 国产视频亚洲色图| 99久久99精品| 亚洲免费黄色| 日韩在线电影一区| 精品视频在线播放一区二区三区| 欧美精品在线极品| 神马午夜一区二区| 欧美性猛交xxxx久久久| 欧美xxxx精品| 国产精品亚洲第一区在线暖暖韩国 | 亚洲国产日韩综合久久精品| 香港三日本8a三级少妇三级99| 免费一区视频| 亚洲蜜桃av| 成人h动漫精品一区二区器材| 91精品国产乱码久久久久久久久| 毛片免费在线| 欧美一卡二卡三卡四卡| 久久露脸国语精品国产91| 国产欧美一区二区三区在线看蜜臀 | 国产成人亚洲综合a∨猫咪| 1024av视频| 久久精品亚洲欧美日韩精品中文字幕| 国产高清在线一区二区| 自拍偷自拍亚洲精品被多人伦好爽| 日韩在线观看免费av| 深爱激情五月婷婷| 欧美日韩激情一区二区| 日本熟女一区二区| 国产精品久久久久久久久图文区| 日本精品一二三| 热久久免费视频| 免费超爽大片黄| 91中文字幕精品永久在线| 国产一区二区三区免费不卡| 男人亚洲天堂| 2020久久国产精品| 亚洲小说区图片| 一区二区在线视频| 天天操天天操天天| 91精品国产综合久久小美女| www欧美在线| 亚洲风情在线资源站| 91免费在线看片| 99久久精品情趣| 亚洲欧美一区二区三区不卡| 日韩精品国产欧美| 黄色av网址在线播放| 婷婷激情综合| 日韩欧美精品一区二区三区经典| 成人av资源网址| 91原创国产| 日韩成人在线一区| 国产精品久久久久久一区二区| a级大胆欧美人体大胆666| 久久久电影免费观看完整版| 福利在线播放| 亚洲欧美在线免费观看| 三级在线观看网站| 欧美tickling挠脚心丨vk| 91片黄在线观看喷潮| 91久久香蕉国产日韩欧美9色| 日韩手机在线观看| 五月天视频一区| 国产精品第72页| 亚洲综合色婷婷| 久久久久97国产| 一区二区三区国产| 青青草原在线免费观看| 亚洲男人的天堂网| 日本精品在线免费观看| 中文字幕一区在线| 日韩在线不卡av| 亚洲女同一区二区| 校园春色 亚洲| 亚洲自拍与偷拍| 精品亚洲永久免费| 亚洲丰满少妇videoshd| 久久夜靖品2区| 午夜电影网亚洲视频| 日韩免费不卡视频| 亚洲午夜久久久久久久久久久| 免费在线黄色片| 亚洲国产精品一区二区久久| 国产亚洲第一页| 精品成人在线视频| 亚洲婷婷综合网| 欧美性大战久久| 国产精品羞羞答答在线| 日韩一区二区三区视频在线观看| 国产ts变态重口人妖hd| 精品成人一区二区三区| 天天色综合av| 在线播放亚洲激情| 美女隐私在线观看| 欧美疯狂做受xxxx高潮| 岛国在线视频网站| 国产成人精品日本亚洲| 日韩欧乱色一区二区三区在线 | 国产精品久久久久桃色tv| 乱老熟女一区二区三区| 亚洲精品成人a在线观看| 久久精品国产亚洲av麻豆色欲| 天天综合色天天| 这里只有久久精品视频| 91超碰这里只有精品国产| 亚洲精品久久久久久久久久| 亚洲精品美女在线观看| 成年人在线视频| 欧美日本高清一区| 中文字幕在线免费观看视频| 国产精品普通话| 99re91这里只有精品| 欧美午夜精品理论片a级大开眼界 欧美午夜精品久久久久免费视 | 日韩亚洲国产精品| 伊人国产在线视频| 成人av手机在线观看| 无码人妻丰满熟妇啪啪欧美| 毛片免费在线播放| 无码av免费一区二区三区试看| 天天干天天干天天操| 欧美日韩视频在线一区二区| 精品国产av鲁一鲁一区| 亚洲精品在线91| 成人直播在线| 国产精品 欧美在线| 在线观看视频一区二区三区 | 精品国产九九九| 亚洲精选一区二区| a黄色片在线观看| 国产成人免费91av在线| 2020国产精品极品色在线观看| 快播亚洲色图| 欧美性色综合| 一级在线免费视频| 99re热这里只有精品视频| 欧美一级特黄高清视频| 欧美日韩在线视频首页| 国产三级自拍视频| 这里只有精品视频| 精品精品导航| 成人免费福利在线| 精品理论电影| 69堂免费视频| 国产不卡视频一区二区三区| 国产黄色大片免费看| 午夜婷婷国产麻豆精品| 国产精品一区二区人人爽| 亚洲天堂日韩电影| 91jq激情在线观看| 96久久精品| 91精品国产91久久久久久黑人| 国产真实乱子伦| 99久久婷婷国产综合精品电影 | 欧美日韩在线播放一区| 深夜福利视频在线免费观看| 精品中文字幕乱| 9999在线精品视频| 亚洲自拍的二区三区| 日本午夜精品视频在线观看| 亚洲精品视频大全| 精品欧美国产一区二区三区| 亚洲精品久久久久久久久久久久久久 | 丝袜a∨在线一区二区三区不卡| 国产精品无码自拍| 一区二区三区在线免费| 99久久久国产精品无码免费| 久久精品美女视频网站| 狠狠久久综合| 久久av秘一区二区三区| 美日韩一级片在线观看| 成人在线手机视频| 五月婷婷综合激情| 日韩欧美电影在线观看| 97人人模人人爽人人喊中文字 | 亚洲国产又黄又爽女人高潮的| 亚洲丝袜一区| 国产一区二区自拍| 国产欧美午夜| 久久美女免费视频| 欧美性色黄大片手机版| 国产一二三在线观看| 国产精品久久久久91| 日韩精品欧美| 中文字幕在线视频一区二区三区 | 国产树林野战在线播放| 国产一区二区三区四区五区入口| 免费在线观看黄色小视频| 欧美一区二区三区视频在线| 日本一本在线免费福利| 成人av免费在线看| 亚洲黄网站黄| 91国内精品久久| 亚洲av永久无码国产精品久久| 欧美巨大黑人极品精男| 一区二区在线免费播放| 久久精品无码中文字幕| 91在线观看免费视频| 波多野结衣激情视频| 最近2019好看的中文字幕免费| 中文字幕综合| 欧美精品卡一卡二| 26uuu国产日韩综合| 在线观看国产精品视频| 欧美成人精品一区二区| 六月丁香久久丫| 亚洲成人福利在线观看| 亚洲激情一二三区| 凸凹人妻人人澡人人添| 国产精品视频久| 欧美日韩一区自拍| 手机免费看av| 日韩视频在线永久播放| 黄色在线免费观看网站| 翔田千里亚洲一二三区| 国产91丝袜在线观看| 亚洲欧美偷拍一区| 欧美www在线| 亚洲婷婷丁香| 国产麻豆剧传媒精品国产| 在线这里只有精品| 在线中文字幕第一页| 欧美精品亚洲| 国产不卡视频一区| 夜夜躁日日躁狠狠久久av| 欧美激情a在线| 日韩综合精品| 久久国产精品无码一级毛片| 欧美日韩高清一区二区| 三妻四妾的电影电视剧在线观看| 一区二区三区av在线| 久久久久久夜精品精品免费|