20250217 隨筆 redis非原子性操作簡述

從你提供的文本來看,核心是 Redis 作為緩存的檢查機制,以及非原子性操作導致的不一致性問題。
我們可以拆解為兩個部分來理解:


📌 1. 邏輯:先查 Redis,再決定是否注冊

邏輯流程

  1. 先查詢 Redis 是否有某個 key(可能是用戶注冊標識、驗證碼等)。
  2. 如果 Redis 中有數據,則直接返回(避免重復操作)。
  3. 如果 Redis 中沒有數據,則執行新的注冊,并存入 Redis,設置60秒過期時間

示例代碼

String key = "user:register:" + phoneNumber;
String cachedValue = redisTemplate.opsForValue().get(key);if (cachedValue == null) {  // Redis 無數據,執行注冊邏輯// 執行新用戶注冊userService.register(phoneNumber);// 將 key 存入 Redis,防止重復注冊redisTemplate.opsForValue().set(key, "registered", 60, TimeUnit.SECONDS);
} else {System.out.println("該用戶已注冊");
}

? 作用

  • 通過 Redis 緩存查詢結果,降低數據庫查詢次數,提高性能。
  • 避免并發重復注冊。

📌 2. 非原子性操作,導致數據不一致

這一部分說的是 多個 Redis 操作不是原子的,可能會導致數據不一致問題,尤其是在并發情況下

? 問題:非原子性操作

Redis 并不自動保證多個操作的原子性。如果有多個線程同時執行:

  1. A 線程查詢 Redis,發現 key 不存在(緩存失效)。
  2. B 線程也查詢 Redis,發現 key 也不存在(同樣緩存失效)。
  3. A 和 B 線程都開始執行注冊邏輯(導致重復注冊)。
  4. A 和 B 線程都將數據寫入 Redis,但只有一個生效,另一個可能會被覆蓋。
🛠 解決方案
  1. 使用 SETNX 實現分布式鎖

    • SETNX(SET if Not eXists)可以確保只有一個線程可以寫入 key,避免并發問題:
    Boolean success = redisTemplate.opsForValue().setIfAbsent(key, "registered", 60, TimeUnit.SECONDS);
    if (Boolean.TRUE.equals(success)) {userService.register(phoneNumber);
    } else {System.out.println("該用戶已注冊");
    }
    

    作用

    • 只有第一個線程能成功寫入 key,其余線程會失敗,避免重復注冊。
  2. 使用 Redis 事務 (MULTI / EXEC)

    • MULTI + EXEC 確保多個 Redis 操作在同一事務中執行
    redisTemplate.execute((RedisCallback<Object>) connection -> {connection.multi();connection.set(key.getBytes(), "registered".getBytes());connection.expire(key.getBytes(), 60);return connection.exec();
    });
    

    作用

    • 確保寫入 key 和設置過期時間是原子的,不會被中途修改。
  3. 使用 Redisson 分布式鎖

    • 適用于高并發場景,保證多個服務器之間不會重復注冊:
    RLock lock = redissonClient.getLock("register-lock:" + phoneNumber);
    try {if (lock.tryLock(5, 10, TimeUnit.SECONDS)) { userService.register(phoneNumber);}
    } finally {lock.unlock();
    }
    

    作用

    • 只有一個線程能拿到鎖,執行注冊,其他線程等待或直接返回。

🔍 結論

  • 先查 Redis,若不存在則執行注冊并緩存數據,提升性能,減少數據庫壓力。
  • 但多個 Redis 操作非原子性,可能會導致數據不一致(如并發下重復注冊)。
  • 解決方案
    1. SETNX(互斥鎖):確保只有一個線程能注冊。
    2. Redis 事務:保證多個 Redis 操作同時執行。
    3. Redisson 分布式鎖:在多服務器環境下防止并發問題。

? 最終目標確保并發情況下數據一致性,避免重復注冊或異常數據覆蓋。 🚀

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/895802.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/895802.shtml
英文地址,請注明出處:http://en.pswp.cn/news/895802.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

git-提交時間和作者時間的區別

1.介紹 定義介紹 提交時間&#xff08;Committer Date&#xff09;&#xff1a;決定了提交在 Git 歷史中的位置&#xff0c;通常影響 GitHub 上提交顯示的順序。 作者時間&#xff08;Author Date&#xff09;&#xff1a;雖然不影響提交的排序&#xff0c;但在每個提交詳情頁…

PHP框架入門指南:從零構建現代Web應用

一、為什么需要PHP框架? 1.1 傳統PHP開發的痛點 重復造輪子:用戶認證、表單驗證等基礎功能需要反復開發代碼混亂:缺乏統一結構導致維護困難安全漏洞:手動處理SQL注入/XSS攻擊效率低下擴展性差:耦合代碼難以適應業務增長1.2 框架的核心價值 標準化架構:MVC模式強制代碼分…

Leetcode 146 LRU緩存 的三種解法

146. LRU 緩存 請你設計并實現一個滿足 LRU (最近最少使用) 緩存 約束的數據結構。 實現 LRUCache 類&#xff1a; LRUCache(int capacity) 以 正整數 作為容量 capacity 初始化 LRU 緩存int get(int key) 如果關鍵字 key 存在于緩存中&#xff0c;則返回關鍵字的值&#xff0…

尚硅谷 java 學習Day19 抽象類與抽象方法、接口、內部類

6-5 抽象類(abstract)與抽象方法&#xff08;important&#xff09; 一、什么叫抽象類&#xff1a; 有時候將一個父類設計的非常抽象&#xff0c;以至于它沒有具體的實例&#xff0c;這樣的類稱為抽象類 abstract關鍵字的使用&#xff1a; ? 1、abstract:抽象的 ? 2、abs…

【LeetCode Hot100 鏈表(上)】相交鏈表、反轉鏈表、回文鏈表、環形鏈表、合并兩個有序鏈表、兩數相加

鏈表 1. 相交鏈表問題描述解決思路代碼實現 2. 反轉鏈表問題描述解決思路代碼實現 3. 回文鏈表問題描述解決思路代碼實現 4. 環形鏈表問題描述解決思路代碼實現 5. 環形鏈表II問題描述解決思路代碼實現 6. 合并兩個有序鏈表問題描述解決思路代碼實現 7. 兩數相加問題描述解決思…

【Python pro】基本數據類型

一、數字類型 1.1 數字類型的組成 1.1.1 整數 &#xff08;1&#xff09;十進制&#xff0c;二進制0b&#xff0c;八進制0o&#xff0c;十六進制0x print(16 0b10000 0o20 0x10) # 輸出&#xff1a;True&#xff08;2&#xff09;十進制轉其他進制 a bin(16) b oct(1…

拯救者電腦在重裝系統之后電源計劃丟失Fn+Q切換不了模式怎么恢復?

參考聯想知識庫的一下鏈接&#xff1a; https://iknow.lenovo.com.cn/detail/196192 其中下載的解壓文件后的文件需要復制粘貼到D盤的根目錄下&#xff0c;再來運行文件。若在生成的log文件中看到導入成功以及控制面板中看到已添加的電源計劃即可 如果還是無效可是試試以下的…

ubuntu 執行 sudo apt-get update 報錯

記錄一下&#xff0c;遇到這個問題了&#xff0c;網絡上看到的解決辦法&#xff0c;親測有效 執行sudo apt-get update ,卻報以下錯誤&#xff0c;“SECURITY: URL redirect target contains control characters rejecting ” 經檢查發現&#xff0c;/etc/apt/source.list 下的…

深度集成DeepSeek大模型:WebSocket流式聊天實現

目錄 5分鐘快速接入DeepSeek大模型&#xff1a;WebSocket實時聊天指南創建應用開發后端代碼 (Python/Node.js)結語 5分鐘快速接入DeepSeek大模型&#xff1a;WebSocket實時聊天指南 創建應用 訪問DeepSeek官網 前往 DeepSeek官網。如果還沒有賬號&#xff0c;需要先注冊一個。…

java斷點調試(debug)

在開發中&#xff0c;新手程序員在查找錯誤時, 這時老程序員就會溫馨提示&#xff0c;可以用斷點調試&#xff0c;一步一步的看源碼執行的過程&#xff0c;從而發現錯誤所在。 重要提示: 斷點調試過程是運行狀態&#xff0c;是以對象的運行類型來執行的 斷點調試介紹 斷點調試是…

軟件技術實訓室解決方案(2025年最新版)

軟件產業作為新興產業的核心組成部分&#xff0c;是推動數字經濟發展的重要力量。在“十四五”規劃的新機遇與挑戰下&#xff0c;我國已明確將加強關鍵數字技術創新應用作為戰略重點&#xff0c;并將軟件和信息技術服務業的發展列為重中之重。這不僅是為了加速構建現代產業體系…

foobar2000設置DSP使用教程及軟件推薦

foobar2000安卓中文版&#xff1a;一款高品質手機音頻播放器 foobar2000安卓中文版是一款備受好評的高品質手機音頻播放器。 幾乎支持所有的音頻格式&#xff0c;包括 MP3、MP4、AAC、CD 音頻等。不論是經典老歌還是最新的流行音樂&#xff0c;foobar2000都能完美播放。除此之…

DeepSeek企業級部署實戰指南:從服務器選型到Dify私有化落地

對于個人開發者或嘗鮮者而言&#xff0c;本地想要部署 DeepSeek 有很多種方案&#xff0c;但是一旦涉及到企業級部署&#xff0c;則步驟將會繁瑣很多。 比如我們的第一步就需要先根據實際業務場景評估出我們到底需要部署什么規格的模型&#xff0c;以及我們所要部署的模型&…

I2C、SPI、UART

I2C&#xff1a;串口通信&#xff0c;同步&#xff0c;半雙工&#xff0c;雙線&#xff08;數據線SDA時鐘線SCL&#xff09;&#xff0c;最大距離1米到幾米 SPI&#xff08;串行外設接口&#xff09;&#xff1a;串口通信&#xff0c;同步&#xff0c;全雙工&#xff0c;四線&…

uniapp 連接mqtt

1&#xff1a;下載插件 npm install mqtt 2&#xff1a;創建 mqtt.js /* main.js 項目主入口注入實例 */ // import mqttTool from ./lib/mqttTool.js // Vue.prototype.$mqttTool mqttTool/* 使用范例見 /pages/index/index.vue */ // mqtt協議&#xff1a;H5使用ws/wss APP-…

shell腳本備份PostgreSQL數據庫和庫下表

注意&#xff1a; 以下為對PostgreSQL13.16版本數據庫備份shell腳本參考請確認備份節點上psql和pgdump的版本不至于太低&#xff0c;建議>13.16該腳本目前是對于整庫、&#xff08;默認針對public這個schema&#xff0c;如果有其他schema&#xff0c;請自行添加一層循環&am…

EXCEL解決IF函數“您已為此函數輸入太多個參數”的報錯

IF函數的基本結構是IF(條件, 值為真時的結果, 值為假時的結果)&#xff0c;所以標準的IF函數最多只能有三個參數。當用戶輸入的參數超過三個時&#xff0c;Excel就會報這個錯誤。比如多個IF語句疊加&#xff0c;但可能在嵌套的過程中沒有正確關閉每個IF函數的括號&#xff0c;導…

圖像質量評價指標-UCIQE-UIQM

一、評價指標UCIQE 在文章《An underwater color image quality evaluation metric》中&#xff0c;提到的了評價指標UCIQE&#xff08;Underwater Colour Image Quality Evaluation&#xff09;&#xff0c;是一種無參考圖像質量評價指標&#xff0c;主要用于評估水下圖像的質…

Vue 前端開發中的路由知識:從入門到精通

文章目錄 引言1. Vue Router 簡介1.1 安裝 Vue Router1.2 配置 Vue Router1.3 在 Vue 實例中使用 Vue Router 2. 路由的基本用法2.1 路由映射2.2 路由視圖2.3 路由鏈接 3. 動態路由3.1 動態路徑參數3.2 訪問動態參數3.3 響應路由參數的變化 4. 嵌套路由4.1 定義嵌套路由4.2 渲染…

基于Springboot+微信小程序調用文心一言大模型實現AI聊天

一、文章前言 此文主要實現基于Springboot微信小程序調用文心一言大模型實現AI聊天對話功能&#xff0c;使用Java作為后端語言進行支持&#xff0c;界面友好&#xff0c;開發簡單。 二、開發流程及工具準備 2.1、登錄百度智能云平臺&#xff0c;獲取 API Key 和 Secret Key兩個…