EF Code 如何應對高併發

来源:https://www.cnblogs.com/wskxy/archive/2023/03/17/17228975.html
-Advertisement-
Play Games

1、高併發的情況,時常會發生數據不穩定的情況 在看本節內容之前,請先看上一章SqlServer 高併發的情況下,如何利用鎖保證數據的穩定性 本節內容,也是具體討論如何在EF中實現這些操作 2、場景模擬,同上一章,搶券 EF 不考慮高併發的情況下,搶券代碼為: string _currOwner = ...


1、高併發的情況,時常會發生數據不穩定的情況

  在看本節內容之前,請先看上一章SqlServer 高併發的情況下,如何利用鎖保證數據的穩定性

  本節內容,也是具體討論如何在EF中實現這些操作

2、場景模擬,同上一章,搶券

  EF 不考慮高併發的情況下,搶券代碼為:

string _currOwner = Console.ReadLine();//當前用戶
using var ctx = new MyDBContext();
var cop = ctx.Coupons.Single(x => x.Id == 2);
if (!string.IsNullOrEmpty(cop.Owner))
{
    Console.WriteLine($"券被搶了");
}
else
{
    cop.Owner = _currOwner;
    Thread.Sleep(5000);
    ctx.SaveChanges();
    Console.WriteLine($"恭喜{_currOwner}搶到券{cop.Id}了");
}
Console.ReadLine();

  打開兩個進程,讓tom和jerry同時先後進行搶券,模擬出一個券同時被兩個用戶搶到的情況

  

  

  上圖可用直觀看出,都提示搶券成功,但是owner是晚一點點執行update的jerry,在實際生產中,無法給tom一個交代

3、解決併發問題

  3.1 通過updlock,悲觀併發控制

string _currOwner = Console.ReadLine();//當前用戶
using var ctx = new MyDBContext();
using var tx = ctx.Database.BeginTransaction();
FormattableString sql = $@"select * from Coupons with(updlock) where id=2";
var cop = ctx.Coupons.FromSqlInterpolated(sql).Single();
if (!string.IsNullOrEmpty(cop.Owner))
{
    Console.WriteLine($"券被搶了");
}
else
{
    cop.Owner = _currOwner;
    Thread.Sleep(5000);
    ctx.SaveChanges();
    Console.WriteLine($"恭喜{_currOwner}搶到券{cop.Id}了");
}
tx.Commit();
Console.ReadLine();

  解決:但這個是排他鎖,有可能造成線程卡頓問題

    

  3.2 通過定義鑒權欄位,樂觀併發控制

    CouponConfig添加配置

            builder.Property(x => x.Owner).IsConcurrencyToken();

    搶券代碼:

string _currOwner = Console.ReadLine();//當前用戶
using var ctx = new MyDBContext();
var cop = ctx.Coupons.Single(x => x.Id == 2);
if (!string.IsNullOrEmpty(cop.Owner))
{
    Console.WriteLine($"券被搶了");
}
else
{
    Thread.Sleep(5000);
    try
    {
        cop.Owner = _currOwner;
        await ctx.SaveChangesAsync();
        Console.WriteLine($"恭喜{_currOwner}搶到券{cop.Id}了");
    }
    catch (DbUpdateConcurrencyException ex)
    {
        var entry = ex.Entries.First();
        var dbValues = entry.GetDatabaseValues();
        var newOwner = dbValues.GetValue<string>(nameof(Coupon.Owner));
        Console.WriteLine($"併發衝突,{newOwner}已經搶到該券了");
    }
}

    結果:

      

    根據update語句,可用看出where加了owner=舊值,來判斷是否發生過更改

  3.3 添加數據版本標識

    如果無法定義一個明確的鑒權欄位,那麼可用通過新增一個欄位,來標識數據來進行鑒權

    public class Coupon
    {
        public int Id { get; set; }
        public string Name { get; set; }
        public string? Description { get; set; }
        public string? Owner { get; set; }
        public byte[] RowVersion { get; set; } #遷移到資料庫,類型為rowversion,當數據更新時,版本會自動遞增
    }

    遷移後資料庫表代碼

CREATE TABLE [dbo].[Coupons] (
    [Id]          INT            IDENTITY (1, 1) NOT NULL,
    [Name]        NVARCHAR (MAX) NOT NULL,
    [Description] NVARCHAR (MAX) NULL,
    [Owner]       NVARCHAR (MAX) NULL,
    [RowVersion]  ROWVERSION     NOT NULL,
    CONSTRAINT [PK_Coupons] PRIMARY KEY CLUSTERED ([Id] ASC)
);

    CouponConfig添加配置

            builder.Property(x => x.RowVersion).IsRowVersion();

    搶券代碼同3.2

    結果:

  

  理論和3.2相同,where會做一個rowversion的舊值判斷

 

  總結:這三種方法由淺入深,各有利弊,在併發量不大的情況下使用3.1,併發量較大的情況下使用3.2&3.3

 

  感謝關註!!

 

 

 

 

 

 

 

 

 

   

 


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

-Advertisement-
Play Games
更多相關文章
  • 前後端分離下EasyExcel的使用 項目環境:SpringBoot+Vue 依賴導入 <!--easyexcel--> <dependency> <groupId>com.alibaba</groupId> <artifactId>easyexcel</artifactId> <version>3 ...
  • 小梅哥的這個ZYNQ開發板上的DDR3位於PS側,PL側想要使用DDR3作為緩存的話,得通過HP介面來與PS側的DDR3控制進行通信。 本次實驗在小梅哥OV5640工程的基礎上,通過修改VDMA的S2MM端的模塊而來的。 將VMDA的幀緩存區設為1,關閉幀同步的功能後,其實和DMA差不多。 一、需要 ...
  • 🎈 開啟多個 ws 服務失敗 正常情況下,如果你想開啟多個 websocket 服務的話 只要在一個文件中,輸入 new Worker 兩次,監聽不同埠,使用 Worker::runAll() 命令即可 但是你會發現在在 windows 中無法在一個文件中同時監聽兩個 websocket 服務, ...
  • 來源:https://liuchenyang0515.blog.csdn.net/article/details/121049426 1. 雙重校驗鎖單例(DCL) public class Singleton { private static volatile Singleton singleto ...
  • 訂單業務一直都是系統研發中的核心模塊,訂單的產生過程,與系統中的很多模塊都會高度關聯,比如賬戶體系、支付中心、運營管理等,即便單看訂單本身,也足夠的複雜; ...
  • 本文介紹了什麼是操作符重載、為什麼需要操作符重載、如何在Java中實現操作符重載以及一些建議。 什麼是操作符重載 操作符重載,就是把已經定義的、有一定功能的操作符進行重新定義,來完成更為細緻具體的運算等功能。從面向對象的角度說,就是可以將操作符定義為類的方法,使得該操作符的功能可以用來代表對象的某個 ...
  • 本文已經收錄到Github倉庫,該倉庫包含電腦基礎、Java基礎、多線程、JVM、資料庫、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分散式、微服務、設計模式、架構、校招社招分享等核心知識點,歡迎star~ Github地址:https://github.c ...
  • 之前,在使用異常捕獲語句try...catch...throw語句時,一直沒太留意幾種用法的區別,前幾天調試程式時發展找不到異常根源,無意中瞭解到幾種使用方法是有區別的。總結如下: 我們都知道,C#中使用throw和throw ex拋出異常,但二者是有區別的。 在C#中推薦使用throw;來拋出異常 ...
一周排行
    -Advertisement-
    Play Games
  • MQTTnet 是一個高性能的MQTT類庫,支持.NET Core和.NET Framework。 MQTTnet 原理: MQTTnet 是一個用於.NET的高性能MQTT類庫,實現了MQTT協議的各個層級,包括連接、會話、發佈/訂閱、QoS(服務質量)等。其原理涉及以下關鍵概念: MqttCli ...
  • 在WPF中,源屬性(Source Property)指的是提供數據的屬性,通常是數據模型或者其他控制項的屬性,而目標屬性(Target Property)則是數據綁定的目標,通常是綁定到控制項的屬性,例如TextBlock的Text屬性。數據綁定將源屬性的值自動更新到目標屬性中。 主要包含以下幾個事件: ...
  • async/await 是 C# 中非同步編程的關鍵特性,它使得非同步代碼編寫更為簡單和直觀。下麵深入詳細描述了 async/await 的使用場景、優點以及一些高級使用方法,並提供了相應的實例源代碼。 使用場景: I/O 操作: 非同步編程特別適用於涉及 I/O 操作(如文件讀寫、網路請求等)的場景。在 ...
  • 使用過office的visio軟體畫圖的小伙伴都知道,畫圖軟體分為兩部分,左側圖形庫,存放各種圖標,右側是一個畫布,將左側圖形庫的圖標控制項拖拽到右側畫布,就會生成一個新的控制項,並且可以自由拖動。那如何在WPF程式中,實現類似的功能呢?今天就以一個簡單的小例子,簡述如何在WPF中實現控制項的拖拽和拖動,... ...
  • 1、Blazor Hybrid簡介 Blazor Hybrid 使開發人員能夠將桌面和移動本機客戶端框架與 .NET 和 Blazor 結合使用。在 Blazor Hybrid 應用中,Razor 組件在設備上是本機運行的。 這些組件通過本地互操作通道呈現到嵌入式 Web 視圖控制項。 組件不在瀏覽器 ...
  • 除了內置的數據集,scikit-learn還提供了隨機樣本的生成器。通過這些生成器函數,可以生成具有特定特性和分佈的隨機數據集,以幫助進行機器學習演算法的研究、測試和比較。 目前,scikit-learn庫(v1.3.0版)中有20個不同的生成樣本的函數。本篇重點介紹其中幾個具有代表性的函數。 1. ...
  • 從0到1,手把手帶你開發截圖工具ScreenCap------002實現通過文件對話框,選擇合適的文件夾,自定義預設的圖片保存位置,簡單易學 ...
  • 每次談到容器的時候,除了Docker之外,都會說起 Kubernetes,那麼什麼是 Kubernetes呢?今天就來一起學快速入門一下 Kubernetes 吧!希望本文對您有所幫助。 Kubernetes,一種用於管理和自動化雲中容器化工作負載的工具。 想象一下你有一個管弦樂隊,將每個音樂家視為 ...
  • 目錄 基本說明 安裝 Nginx 部署 VUE 前端 部署 Django 後端 Django admin 靜態文件(CSS,JS等)丟失的問題 總結 1. 基本說明 本文介紹了在 windows 伺服器下,通過 Nginx 部署 VUE + Django 前後端分離項目。本項目前端運行在 80 埠 ...
  • 從0到1,手把手帶你開發截圖工具ScreenCap------003實現最小化程式到托盤運行,- 為了方便截圖乾凈,實現最小化程式到托盤運行,簡潔,勿擾,實現最小化程式到托盤運行, 實現托盤菜單功能,實現回顯主窗體, 實現托盤開始截屏, 實現氣泡信息提示,實現托盤程式提示,實現托盤退出程式, 封裝完... ...