【Redis】Redis事務:原子性與回滾的真相揭秘

来源:https://www.cnblogs.com/maerpao/archive/2023/09/25/17705258.html
-Advertisement-
Play Games

大家好,我是mep。今天一起來探討一下Redis緩存的問題,SpringBoot如何集成Redis網上文章很多,基本都是介紹如何配置redisTemplate,如何調用,本文就不過多介紹了。這次我們研究的是:Redis的事務。 首先拋出一個問題,Redis支持事務嗎? 答案肯定是支持,不然也不需要我 ...


大家好,我是mep。今天一起來探討一下Redis緩存的問題,SpringBoot如何集成Redis網上文章很多,基本都是介紹如何配置redisTemplate,如何調用,本文就不過多介紹了。這次我們研究的是:Redis的事務。

首先拋出一個問題,Redis支持事務嗎?

 

 

答案肯定是支持,不然也不需要我們在這裡探討了。

然後你拿到關鍵詞"Redis 事務"去搜索引擎搜索一下,得到了這樣的答案:

Redis支持事務,But!Redis的事務不保證原子性,事務不會回滾。例如:我在Redis中提交了一個事務,包含3條命令,其中第2條命令報錯了,並不會導致第一條命令的回滾,也不會阻止第三條命令的執行。

可是,真的是這樣嗎?你試過嗎?哈哈,知道你懶得試,我來幫你們試試看嘍!

先看一個我自己測試的例子,以下例子中RedisTemplate都開啟了事務支持,否則測試沒有意義,我的RedisConfiguration代碼如下:

@Configuration
public class RedisConfiguration {
    @Bean
    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {
        RedisTemplate<String, Object> template = new RedisTemplate<>();
        template.setEnableTransactionSupport(true);
        template.setConnectionFactory(factory);
        template.setKeySerializer(new StringRedisSerializer());
        template.setHashKeySerializer(new StringRedisSerializer());
        template.setValueSerializer(new GenericJackson2JsonRedisSerializer());
        template.setHashValueSerializer(new GenericJackson2JsonRedisSerializer());
        template.afterPropertiesSet();
        return template;
    }
}

 

例1,使用@Transactional註解,方法執行過程中報錯,代碼如下:

    @Transactional
    public void testRedisTransaction() {
        employeeMapper.updateByPrimaryKey(Employee.builder()
                        .id(4L)
                        .name("uuuuu")
                        .gender(Gender.MALE)
                .build());
        redisTemplate.opsForValue().set("111", "111");
        int i = 1 / 0;
        redisTemplate.opsForValue().set("222", "111");
    }

執行前Redis緩存情況:

執行以上方法後,肯定會報錯:

java.lang.ArithmeticException: / by zero

猜猜執行完後資料庫和Redis中數據操作是什麼樣的?

 

1.資料庫會回滾,即update無效,這個並不意外,沒啥可說的

2.執行後Redis緩存情況:

 空的?不是說Redis的事務不支持回滾嗎?為什麼key修改卻無效了呢?

確實,因為Redis根本沒有回滾,它的事務壓根就沒有提交!!!

這就是Redis的事務和關係型資料庫不一樣的地方,資料庫一個事務中如果某一條SQL報錯或方法中有RuntimeException(@Transactional預設)拋出的話,事務會回滾。對於Redis的事務來說,如果方法中拋RuntimeException的話,事務壓根不提交,被DISCARD之後,自然不會執行。

如果你看到這裡了,說明你一開始就質疑最上面搜索到的結果,連查到的知識都會質疑和驗證,為什麼要相信我上面說的事務壓根就沒有提交的結論呢?

帶著疑問,我們繼續驗證,先上代碼:

    @Transactional
    public void testRedisTransaction() {
        employeeMapper.updateByPrimaryKey(Employee.builder()
                .id(4L)
                .name("uuuuu")
                .gender(Gender.MALE)
                .build());
        System.out.println(1234);
        redisTemplate.opsForValue().set("111", "a");
        redisTemplate.opsForValue().set("222", "a");
        redisTemplate.exec();
        int i = 1 / 0;
    }

這次主動在報異常前提交了Redis事務,結果如下:

 到這裡,我們得到結論是這樣的:

Redis事務不能回滾,方法報異常時事務並沒有回滾,之所以數據沒有被寫入到Redis,是因為事務被DISCARD了

根據我們查到的內容,還需要驗證Redis的事務不能保證原子性,繼續上示例: 

例2,使用@Transactional註解,在Redis事務中報錯,代碼如下:

    @Transactional
    public void testRedisTransactionOnly() {
        redisTemplate.opsForValue().set("333", "a");
        redisTemplate.opsForHash().put("333", "a", 111);
    }

正常來說,應該會報WRONGTYPE Operation的錯誤,不過,執行結果是這樣的:

 甚至,連個錯誤都沒有報!

是代碼的問題嗎?還是因為Redis的事務忽略了異常的命令,只執行了正常的命令?

繼續測試,清空Redis,去掉@Transactional註解:

//    @Transactional
    public void testRedisTransactionOnly() {
        redisTemplate.opsForValue().set("333", "a");
        redisTemplate.opsForHash().put("333", "a", 111);
    }

執行結果:

 可見代碼沒有問題,確實會報錯,只是提交到一個事務中,它不保證原子性,只執行了可執行的命令,即使後續的命令報錯,也不會回滾,而且不會報錯

至此,Redis事務相關的驗證已結束。

結論就是我們開始搜索到的結果:

Redis支持事務,But!Redis的事務不保證原子性,事務不會回滾,提交後會執行可正常執行的命令,忽略報錯的命令。

最後,來自Redis官網的一句話佐證我們的結論, 附出處:Transactions | Redis


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 近幾年來Laravel在PHP領域大放異彩,逐漸成為PHP開發框架中的中流砥柱。 這個系列的文章, 會帶你一起探知Laravel框架底層的實現細節。與其他框架相比,Laravel的設計理念確實更為先進(服務、容器、依賴註入、facade。。。),初讀代碼時會感覺代碼晦澀難懂,而一旦弄清了整套框架的基 ...
  • 1 摘要 通過使用記錄模式來增強Java編程語言,以解構記錄值。記錄模式和類型模式可嵌套使用,從而實現強大、聲明式和可組合的數據導航和處理形式。 2 發展史 由 JEP 405 提出的預覽功能,併在JDK 19發佈,然後由 JEP 432 再次預覽,併在JDK 20發佈。該功能與用於switch的模 ...
  • 來源:麥叔編程 作者:小K 前言 一個好的變數名能讓讀代碼的人(包括寫的人),身心舒暢,但一個“奇葩”的變數名可能會逼瘋一個程式員。 今天是奇葩變數名大賞! 正文 註:以下素材均採集自網路 先上場的是某企業機房的門牌: 我猜這個主任可能是個胡建人。 推薦一個開源免費的 Spring Boot 實戰項 ...
  • 一 背景 C端服務應用升級和重啟,導致耗時瞬時抖動,業務超時,應用監控報警,上游感知明顯,導致用戶體驗變差。 二 應用升級重啟導致抖動的原因 1 C端服務應用升級和重啟的冷啟動階段,它需要重新載入和初始化各種資源,例如資料庫連接、緩存數據等,導致耗時瞬時飆升。 2 應用重啟後,本地緩存失效,應用需要 ...
  • 1 全新併發編程模式 JDK9 後的版本你覺得沒必要折騰,我也認可,但是JDK21有必要關註。因為 JDK21 引入全新的併發編程模式。 一直沽名釣譽的GoLang吹得最厲害的就是協程了。JDK21 中就在這方面做了很大的改進,讓Java併發編程變得更簡單一點,更絲滑一點。 之前寫過JDK21 Fe ...
  • 今天不知為何開始報錯 Entry WEB-INF/classes/classpath.index is a duplicate but no duplicate handling strategy has been set.,大約是由於 我把 Gradle 遷移到了 Kotlin 導致的 經過一番搜 ...
  • 大家好,我是TJ君! 如今在國內運營的各種互聯網應用都有接入IP來源顯示的要求,現在相關API的供應商也很多。今天TJ剛好看到一個不錯的,所以馬上給大家推薦一下。 這款不錯的產品名稱為:IPInfo 產品特性 該IP查詢工具除了傳統的提供地址位置之外,還有很多其他能力,具體的這裡TJ君給大家整理了一 ...
  • 前兩天一個小伙伴突然找我求助,說準備換個坑,最近在系統複習多線程知識,但遇到了一個刷新認知的問題…… ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...