Redis修改數據多線程併發—Redis併發鎖

来源:http://www.cnblogs.com/tdws/archive/2016/07/27/5712835.html
-Advertisement-
Play Games

轉載爬蟲請註明地址,博客園蝸牛 http://www.cnblogs.com/tdws/p/5712835.html 先配上一個簡易的RedisHelper,一個set值,一個get值,一個設置併發鎖,以便在我後面的操作中,你能清楚我究竟做了什麼。 1 public class RedisHelpe ...


轉載爬蟲請註明地址,博客園蝸牛 http://www.cnblogs.com/tdws/p/5712835.html

先配上一個簡易的RedisHelper,一個set值,一個get值,一個設置併發鎖,以便在我後面的操作中,你能清楚我究竟做了什麼。

 1  public class RedisHelper
 2     {
 3         public RedisClient client = new RedisClient("127.0.0.1", 6379);
 4         public void Set<T>(string key, T val)
 5         {
 6             client.Set(key, val);
 7         }
 8         public T Get<T>(string key)
 9         {
10             var result = client.Get<T>(key);
11             return result;
12         }
13         public IDisposable Setnx(string key)
14         {
15            return  client.AcquireLock(key);
16         }
17     }
View Code

下麵看一下併發代碼,我只new了兩個Thread。兩個線程同時想訪問同一個key,分別訪問五萬次,在併發條件下,我們很難保證數據的準確性,請比較輸出結果。

 1 static void Main(string[] args)
 2         {
 3             RedisHelper rds = new RedisHelper();
 4             rds.Set<int>("mykey1", 0);
 5             Thread myThread1 = new Thread(AddVal);
 6             Thread myThread2 = new Thread(AddVal);
 7             myThread1.Start();
 8             myThread2.Start();
 9             Console.WriteLine("等待兩個線程結束");
10             Console.ReadKey();
11         }
12 
13         public static void AddVal()
14         {
15             RedisHelper rds = new RedisHelper();
16             for (int i = 0; i < 50000; i++)
17             {
18                 
19                     int result = rds.Get<int>("mykey1");
20                     rds.Set<int>("mykey1", result + 1);
21                 
22             }
23             Console.WriteLine("線程結束,輸出" + rds.Get<int>("mykey1"));
24         }
View Code

是的,和我們單線程,跑兩個50000,會輸出100000。現在是兩個併發線程同時跑在由於併發造成的數據結果往往不是我們想要的。那麼如何解決這個問題呢,Redis已經為我們準備好了!

你可以看到我RedisHelper中有個方法是 public IDisposable Setnx(string key)。  也可以看到他返回的是IDisposable,證明我們需要手動釋放資源。方法內部的 AcquireLock正是關鍵之處,它像redis中索取一把鎖頭,被鎖住的資源,只能被單個線程訪問,不會被兩個線程同時get或者set,這兩個線程一定是交替著進行的,當然這裡的交替並不是指你一次我一次,也可能是你多次,我一次,下麵看代碼。

 1  static void Main(string[] args)
 2         {
 3             RedisHelper rds = new RedisHelper();
 4             rds.Set<int>("mykey1", 0);
 5             Thread myThread1 = new Thread(AddVal);
 6             Thread myThread2 = new Thread(AddVal);
 7             myThread1.Start();
 8             myThread2.Start();
 9             Console.WriteLine("等待兩個線程結束");
10             Console.ReadKey();
11         }
12 
13         public static void AddVal()
14         {
15             RedisHelper rds = new RedisHelper();
16             for (int i = 0; i < 50000; i++)
17             {
18                 using (rds.Setnx("lock"))
19                 {
20                     int result = rds.Get<int>("mykey1");
21                     rds.Set<int>("mykey1", result + 1);
22                 }
23             }
24             Console.WriteLine("線程結束,輸出" + rds.Get<int>("mykey1"));
25         }
View Code

可以看到我使用了using,調用我的Setnx方法獲取鎖。

輸出結果最後是100000,正是我們要的正確結果。前面的8W+是因為兩個線程之一先執行結束了。

還有,在正式使用的過程中,建議給我們的鎖,使用後刪除掉,並加上一個過期時間,使用expire。

以免程式執行期間意外退出,導致鎖一直存在,今後可能無法更新或者獲取此被鎖住的數據。

你也可以嘗試一下不設置expire,在程式剛開始執行時,關閉console,重新運行程式,並且在redis-cli的操作控制台,get你鎖住的值,將會永遠獲取不到。

 


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

-Advertisement-
Play Games
更多相關文章
  • 昨天給大家介紹了實體FluentValidation驗證,今天繼續給大家介紹表單驗證,在Rookey.Frame框架中,表單驗證有PrimaryKeyFields欄位驗證、唯一驗證、必填驗證、常用驗證及自定義驗證,PrimaryKeyFields欄位驗證昨天也略微介紹了下PrimaryKeyFiel ...
  • 數組:具有相同類型的若幹變數按有序的形式組織起來的一種形式。這些按序排列的同類數據元素的集合稱為數組。定義數組 int[] 變數名 = new int [n]; string[] myStringArray = new string[6]; int[] myArray = new int[] {1, ...
  • 現在是一個信息時代,並且正在高速發展.以前獲取信息的途徑非常少,可能只有電視臺,收音機等有限的來源,而現在的途徑數不勝數,如:QQ,微信,官方網站,個人網站等等 本開發手冊是基於聚合數據編寫開發的,其作用是將聚合數據開放平臺中的API能夠使用,測試和展示,方便開發員快速開發.把更多的精力放在業務上... ...
  • 現在是一個信息時代,並且正在高速發展.以前獲取信息的途徑非常少,可能只有電視臺,收音機等有限的來源,而現在的途徑數不勝數,如:QQ,微信,官方網站,個人網站等等 本開發手冊是基於騰訊地圖LBS開放平臺編寫開發的,其作用是將騰訊LBS數據開放平臺中的API能夠使用,測試和展示,方便開發員快速開發.把... ...
  • 因為項目需要,需要在網頁上實現消息的推送。在百度上搜索了一下,發現實現網頁上的消息推送,可以使用asp.net 中的SignalR類庫,當然也可以使用H5的WebSocket Ajax的輪迴。當然此處我們使用asp.net 中的SignalR類庫。因為它可以實現網頁上消息的實時推送。什麼是實時推送呢 ...
  • 說明:本文是個人翻譯文章,由於個人水平有限,有不對的地方請大家幫忙更正。 原文: "dotnet command" 翻譯: "dotnet 命令" 名稱 dotnet 運行命令行命令的一般驅動程式 概要 描述 是命令行界面(CLI)工具鏈的通用驅動程式。調用它自己,會給出簡短的使用說明。 每個特定的 ...
  • 能採集任何網站的各種格式圖片,實現把所有文章、新聞、帖子等中間的圖片全部有有序列的分類後保存到您的電腦上等功能,可以把任何論壇網站的所有帖子的圖片採集到本地,輕鬆過濾廣告等,是網站、論壇站長和喜歡收集美圖的朋友的必備工具。 本文演示針對上一個版本增加這些功能,全站下載、顯示可採集網站、軟體註冊、軟 ...
  • 現在是一個信息時代,並且正在高速發展.以前獲取信息的途徑非常少,可能只有電視臺,收音機等有限的來源,而現在的途徑數不勝數,如:QQ,微信,官方網站,個人網站等等 本開發手冊是基於百度數據開放平臺編寫開發的,其作用是調用百度數據開放平臺中的API方法,方便查找問題,從而實現開發員快速開發.把更多的精力... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...