主頁 > 後端開發 > 學習筆記-Spring事務

學習筆記-Spring事務

2023-05-24 07:45:32 後端開發

學習的文章

小姐姐非要問我:spring編程式事務是啥? (qq.com)

一文搞懂什么是事務 - 知乎 (zhihu.com)

阿里3面:Spring宣告式事務連環炮,讓我措手不及,, (qq.com)

帶你讀懂Spring 事務——事務的傳播機制 - 知乎 (zhihu.com)

spring 事務失效的 12 種場景_事務什么時候失效_hanjq_code的博客-CSDN博客

什么是事務

  • 事務是并發操作的單位

  • 是用戶定義的操作序列

  • 事務如果成功,則會提交

  • 事務如果失敗,則會回滾

事務的四大特性

  • 原子性

    • 事務中操作,要么不做,要么都做
  • 持久性

    • 一個事務一旦提交,它對資料庫的改變是永久的
  • 一致性

    • 事務讓資料庫從一個一致性狀態轉移到另一個一致性狀態

    • 比如

      • 事務前,A有50,B有50,總共有100

      • 事務中,A送給B20

      • 事務后,A有30,B有70,總共還是有100

  • 隔離性

    • 一個事務的執行不能被其他事務所干擾

    • 分成不同的等級

事務并發訪問導致的資料問題

臟讀

讀到了修改但還沒有提交的資料

  • A事務修改了某條記錄的欄位c,但還沒有提交

  • B事務在此時讀取了欄位c

  • A事務發生了回滾,欄位c恢復了修改前的狀態

  • 但B事務持有的還是修改后的狀態

不可重復讀

某個事務在執行程序中,兩次讀同一個條記錄,但結果不一樣

  • A事務讀取了記錄c,值為10

  • B事務修改了記錄c,值為0

  • A事務再次讀取記錄c,值為0

幻讀

某個事務在執行程序中,前后兩次讀取記錄,資料總量不一樣

  • A事務統計了表c中的記錄總數,結果為10

  • B事務洗掉了表c中的5條記錄

  • A事務再次統計了表c中的記錄總數,結果為5

和不可重復讀的區別在于,幻讀針對的是記錄條數,不可重復讀針對的是記錄內容

事務的隔離級別

針對事務并發訪問時出現的問題,設定了四種事務的隔離級別

讀未提交

可以的讀取還沒有提交的資料

沒有限制,三種問題都有可能發生

讀已提交

只能讀取已經提交了的資料

不會發生臟讀,但是會發生不可重復讀和幻讀

可重復讀

一個事務前后讀取的同一條記錄的結果必須一致

不會發生臟讀和不可重復讀,但會發生幻讀

mysql中默認的事務隔離級別

串行化

所有的事務必須依次執行

不會發生臟讀、不可重讀讀和幻讀

效率比較低

Spring事務的使用方法

Spring分為兩種控制事務的方法

  • 編程式事務

    • 方法1:通過PlatformTransactionManager控制事務

    • 方法2:通過TransactionTemplate控制事務

  • 宣告式事務

    • 常用

編程式事務的使用

使用PlatformTransactionManager

@Test
public void test1() throws Exception {
    //定義一個資料源
    org.apache.tomcat.jdbc.pool.DataSource dataSource = new org.apache.tomcat.jdbc.pool.DataSource();
    dataSource.setDriverClassName("com.mysql.jdbc.Driver");
    dataSource.setUrl("jdbc:mysql://localhost:3306/javacode2018?characterEncoding=UTF-8");
    dataSource.setUsername("root");
    dataSource.setPassword("root123");
    dataSource.setInitialSize(5);
    //定義一個JdbcTemplate,用來方便執行資料庫增刪改查
    JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);
    //1.定義事務管理器,給其指定一個資料源(可以把事務管理器想象為一個人,這個人來負責事務的控制操作)
    PlatformTransactionManager platformTransactionManager = new DataSourceTransactionManager(dataSource);
    //2.定義事務屬性:TransactionDefinition,TransactionDefinition可以用來配置事務的屬性資訊,比如事務隔離級別、事務超時時間、事務傳播方式、是否是只讀事務等等,
    TransactionDefinition transactionDefinition = new DefaultTransactionDefinition();
    //3.開啟事務:呼叫platformTransactionManager.getTransaction開啟事務操作,得到事務狀態(TransactionStatus)物件
    TransactionStatus transactionStatus = platformTransactionManager.getTransaction(transactionDefinition);
    //4.執行業務操作,下面就執行2個插入操作
    try {
        System.out.println("before:" + jdbcTemplate.queryForList("SELECT * from t_user"));
        jdbcTemplate.update("insert into t_user (name) values (?)", "test1-1");
        jdbcTemplate.update("insert into t_user (name) values (?)", "test1-2");
        //5.提交事務:platformTransactionManager.commit
        platformTransactionManager.commit(transactionStatus);
    } catch (Exception e) {
        //6.回滾事務:platformTransactionManager.rollback
        platformTransactionManager.rollback(transactionStatus);
    }
    System.out.println("after:" + jdbcTemplate.queryForList("SELECT * from t_user"));
}

使用TransactionTemplate

@Test
public void test1() throws Exception {
    //定義一個資料源
    org.apache.tomcat.jdbc.pool.DataSource dataSource = new org.apache.tomcat.jdbc.pool.DataSource();
    dataSource.setDriverClassName("com.mysql.jdbc.Driver");
    dataSource.setUrl("jdbc:mysql://localhost:3306/javacode2018?characterEncoding=UTF-8");
    dataSource.setUsername("root");
    dataSource.setPassword("root123");
    dataSource.setInitialSize(5);
    //定義一個JdbcTemplate,用來方便執行資料庫增刪改查
    JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);
    //1.定義事務管理器,給其指定一個資料源(可以把事務管理器想象為一個人,這個人來負責事務的控制操作)
    PlatformTransactionManager platformTransactionManager = new DataSourceTransactionManager(dataSource);
    //2.定義事務屬性:TransactionDefinition,TransactionDefinition可以用來配置事務的屬性資訊,比如事務隔離級別、事務超時時間、事務傳播方式、是否是只讀事務等等,
    DefaultTransactionDefinition transactionDefinition = new DefaultTransactionDefinition();
    transactionDefinition.setTimeout(10);//如:設定超時時間10s
    //3.創建TransactionTemplate物件
    TransactionTemplate transactionTemplate = new TransactionTemplate(platformTransactionManager, transactionDefinition);
    /**
     * 4.通過TransactionTemplate提供的方法執行業務操作
     * 主要有2個方法:
     * (1).executeWithoutResult(Consumer<TransactionStatus> action):沒有回傳值的,需傳遞一個Consumer物件,在accept方法中做業務操作
     * (2).<T> T execute(TransactionCallback<T> action):有回傳值的,需要傳遞一個TransactionCallback物件,在doInTransaction方法中做業務操作
     * 呼叫execute方法或者executeWithoutResult方法執行完畢之后,事務管理器會自動提交事務或者回滾事務,
     * 那么什么時候事務會回滾,有2種方式:
     * (1)transactionStatus.setRollbackOnly();將事務狀態標注為回滾狀態
     * (2)execute方法或者executeWithoutResult方法內部拋出例外
     * 什么時候事務會提交?
     * 方法沒有例外 && 未呼叫過transactionStatus.setRollbackOnly();
     */
    transactionTemplate.executeWithoutResult(new Consumer<TransactionStatus>() {
        @Override
        public void accept(TransactionStatus transactionStatus) {
            jdbcTemplate.update("insert into t_user (name) values (?)", "transactionTemplate-1");
            jdbcTemplate.update("insert into t_user (name) values (?)", "transactionTemplate-2");

        }
    });
    System.out.println("after:" + jdbcTemplate.queryForList("SELECT * from t_user"));
}

宣告式事務的使用

  • 在配置類上使用@EnableTransactionManagement

    • Springboot可以加在啟動類上
  • 定義事務管理器

    • @Bean
      public PlatformTransactionManager transactionManager(DataSource dataSource) {
          return new DataSourceTransactionManager(dataSource);
      }
      
    • springboot中有默認的事務管理器

  • 在需要事務的目標上加上@Transaction注解

    • 作用位置

      • 注意:@Transaction只對public方法有效

      • @Transacion放在介面上,介面的所有實作類中所有的public方法都自動加上事務

      • @Transaction放在類上,當前類以及其下無限級子類中的public方法都被加上事務

      • @Transaction放在public方法上,方法被加上事務

    • 屬性

      • "transactionManager"或"value"

        • 指定事務管理器的bean物件

        • 為空的話,默認按型別獲取

      • “propagation”

        • 指定事務的傳播型別

        • 默認為REQUIRED

      • “rollbackFor”

        • 自定義回滾例外

事務的傳播型別

  • REQUIRED

    • 如果當前有事務,則加入當前事務

    • 如果當前沒有事務,則自己新建一個事務

  • SUPPORTS

    • 如果當前有事務,則加入當前事務

    • 如果當前沒有事務,則以非事務方式執行

  • MANDATORY

    • 如果當前有事務,則加入當前事務

    • 如果當前沒有事務,則拋出例外

  • REQUIERES_NEW

    • 如果當前有事務,則將該事務掛起,另外新創建一個事務

    • 如果當前沒有事務,則新創建一個事務

  • NOT_SUPPORTED

    • 如果當前有事務,則將該事務掛起,以非事務方式執行

    • 如果當前沒有事務,則以非事務方式執行

  • NEVER

    • 如果當前有事務,則拋出例外

    • 如果當前沒有事務,以非事務方式執行

  • NESTED

    • 如果當前有事務,則在當前事務中嵌套一個事務執行

    • 如果當前沒有事務,則新建一個事務執行

事務失效或回滾例外的12種情況

事務失效

  • 訪問權限問題

    • 事務只能對public的方法方法生效
  • 方法用final或static修飾

    • spring事務是使用動態代理的方式實作的

    • 如果加了final或public方法,則方法無法被代理

  • 方法內部呼叫

    • @Service
      public class UserService {
       
          @Autowired
          private UserMapper userMapper;
       
        
          public void add(UserModel userModel) {
              userMapper.insertUser(userModel);
              updateStatus(userModel);
          }
       
          @Transactional
          public void updateStatus(UserModel userModel) {
              doSameThing();
          }
      }
      
    • 在add方法中是通過this來呼叫updateStatus方法的,沒有通過代理

    • 解決方法:

      • 1.注入自己

        • @Servcie
          public class ServiceA {
             @Autowired
             prvate ServiceA serviceA;
           
             public void save(User user) {
                   queryData1();
                   queryData2();
                   serviceA.doSave(user);
             }
           
             @Transactional(rollbackFor=Exception.class)
             public void doSave(User user) {
                 addData1();
                 updateData2();
              }
           }
          
      • 2.通過AopContext.currentProxy()獲取代理物件

        • @Servcie
          public class ServiceA {
           
             public void save(User user) {
                   queryData1();
                   queryData2();
                   ((ServiceA)AopContext.currentProxy()).doSave(user);
             }
           
             @Transactional(rollbackFor=Exception.class)
             public void doSave(User user) {
                 addData1();
                 updateData2();
              }
           }
          
  • 未被spring管理

    • 忘了給類添加注釋了

    • 沒有被放到IOC容器中

  • 多執行緒呼叫

    • spring事務是通過資料庫連接來實作的

    • 在多執行緒中,每一個執行緒用的都是不同的資料庫連接

  • 表不支持事務

    • MyISAM引擎的表不支持事務
  • 未開啟事務

    • springboot需要在啟動類上加上@EnableTransactionManagement的注解

    • 傳統spring專案需要在applicationContext.xml中進行相關的配置

事務回滾例外

  • 使用了錯誤的傳播特性

  • 手動捕獲了例外

    • 如果想要spring事務能夠正常回滾,必須拋出它能夠處理的例外

    • 使用try/catch將例外捕獲,將導致事務不會回滾

  • 拋得例外型別不正確

    • spring事務,默認情況下只會回滾RuntimeException或Error
  • 自定義回滾例外不匹配

    • 比如定義了BusinessException,但拋出的是SqlException
  • 嵌套事務回滾范圍多了

    • public class UserService {
       
          @Autowired
          private UserMapper userMapper;
       
          @Autowired
          private RoleService roleService;
       
          @Transactional
          public void add(UserModel userModel) throws Exception {
              userMapper.insertUser(userModel);
              roleService.doOtherThing();
          }
      }
       
      @Service
      public class RoleService {
       
          @Transactional(propagation = Propagation.NESTED)
          public void doOtherThing() {
              System.out.println("保存role表資料");
          }
      }
      
    • roleService.doOtherThing()如果發生回滾,它拋出的例外沒有被處理,會繼續往上級拋

    • add()在捕獲到向上拋的例外后也會發生回滾

    • 應該手動進行捕獲

      • @Slf4j
        @Service
        public class UserService {
         
            @Autowired
            private UserMapper userMapper;
         
            @Autowired
            private RoleService roleService;
         
            @Transactional
            public void add(UserModel userModel) throws Exception {
         
                userMapper.insertUser(userModel);
                try {
                    roleService.doOtherThing();
                } catch (Exception e) {
                    log.error(e.getMessage(), e);
                }
            }
        }
        

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/553187.html

標籤:Java

上一篇:【2023.03.20】P4710 「物理」平拋運動

下一篇:返回列表

標籤雲
其他(159545) Python(38162) JavaScript(25444) Java(18102) C(15231) 區塊鏈(8267) C#(7972) AI(7469) 爪哇(7425) MySQL(7207) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5871) 数组(5741) R(5409) Linux(5340) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4575) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2433) ASP.NET(2403) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1976) 功能(1967) Web開發(1951) HtmlCss(1941) C++(1922) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1878) .NETCore(1861) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • 學習筆記-Spring事務

    學習的文章 [小姐姐非要問我:spring編程式事務是啥? (qq.com)](https://mp.weixin.qq.com/s?__biz=MzA5MTkxMDQ4MQ==&mid=2648936779&idx=2&sn=a6255c7d436a62af380dfa6b326fd4e7&chk ......

    uj5u.com 2023-05-24 07:45:32 more
  • 【2023.03.20】P4710 「物理」平拋運動

    題目傳送門: >[【洛谷】P4710 [物理]平拋運動](https://www.luogu.com.cn/problem/P4710 "【洛谷】P4710 [物理]平拋運動") ## Step 1:前置芝士 您需要知道并了解以下芝士: 1. 數學: - 三角函式; 2. 物理: - 加速度公式; ......

    uj5u.com 2023-05-24 07:44:42 more
  • 驅動開發:內核實作行程匯編與反匯編

    在筆者上一篇文章`《驅動開發:內核MDL讀寫行程記憶體》`簡單介紹了如何通過MDL映射的方式實作行程讀寫操作,本章將通過如上案例實作遠程行程反匯編功能,此類功能也是ARK工具中最常見的功能之一,通常此類功能的實作分為兩部分,內核部分只負責讀寫位元組集,應用層部分則配合反匯編引擎對位元組集進行解碼,此處我們... ......

    uj5u.com 2023-05-24 07:44:33 more
  • 大家聽過Java applet嗎?為什么不再流行了

    Java applet 不知道有同學聽過嗎?我也只是聽過,并沒有使用過。我特意去了解了一下它,本文就對 Java applet 進行簡單介紹,說說它的輝煌與衰敗。僅此而已,現在已經沒人使用 Java applet 開發了。 ......

    uj5u.com 2023-05-23 11:24:08 more
  • 了解JAVA記憶體模型(JMM)

    ## 1、概述 我們常說的JMM指的是Java記憶體模型(Java Memory Model,JMM),主要用于控制Java程式解決執行緒間如何通信和資料同步,JMM規范了多執行緒訪問共享記憶體時的 **可見性、有序性和原子性**。 - 所有的共享變數都存在**主記憶體**中; - **每個執行緒**都保存了一 ......

    uj5u.com 2023-05-23 11:24:04 more
  • JAVA-02(語法介紹)

    # JAVA-02 # Java流程控制 ### 用戶互動Scanner #### Scanner類 **屬于IO流的類如果不關會一直占用資源** ##### 常用方法 ? 【Scanner scanner = new Scanner(System.in);】 ? .next(); 遇到**空格** ......

    uj5u.com 2023-05-23 11:23:57 more
  • 解決啟動jar包報錯:錯誤找不到或無法加載主類 jar

    #### 錯誤: 找不到或無法加載主類 jar ##### 問題描述: 在使用springboot框架對專案打包后,手動使用命令java -jar 包名啟動jar包,報錯:錯誤: 找不到或無法加載主類 jar。 網上找了各辦法,都是加maven插件,打成可執行jar包 ``` org.springf ......

    uj5u.com 2023-05-23 11:23:47 more
  • 用chatGPT快速開發java后端功能

    接到一個緊急需求如圖 常規無非是建表,寫介面,寫測驗類,最后造資料進行自測。突發奇想,要不用GPT4試一下快速寫業務代碼? 寫句子1分鐘,建表和得到代碼1分鐘 第一步:建表,直接復制excel中內容到GPT中 第二步:要求轉為下劃線: 第三步:給出條件和想要的結果(下面第一幅圖是自己寫的,其它代碼圖 ......

    uj5u.com 2023-05-23 11:23:30 more
  • 【重學C++】04 | 說透C++右值參考、移動語意、完美轉發(上)

    ## 文章首發 [【重學C++】04 | 說透C++右值參考、移動語意、完美轉發(上)](https://mp.weixin.qq.com/s/35Jbt-vroWhxTk0SSyhgSQ) ## 引言 大家好,我是只講技術干貨的會玩code,今天是【重學C++】的第四講,在前面《[03 | 手擼C ......

    uj5u.com 2023-05-23 11:23:18 more
  • 大家聽過Java applet嗎?為什么不再流行了

    Java applet 不知道有同學聽過嗎?我也只是聽過,并沒有使用過。我特意去了解了一下它,本文就對 Java applet 進行簡單介紹,說說它的輝煌與衰敗。僅此而已,現在已經沒人使用 Java applet 開發了。 ......

    uj5u.com 2023-05-23 11:22:17 more