亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 數據庫 > SQL Server > 正文

SQL Server并發處理存在就更新解決方案探討

2024-08-31 01:04:55
字體:
來源:轉載
供稿:網友

前言

本節我們來講講并發中最常見的情況存在即更新,在并發中若未存在行記錄則插入,此時未處理好極容易出現插入重復鍵情況,本文我們來介紹對并發中存在就更新行記錄的七種方案并且我們來綜合分析最合適的解決方案。

探討存在就更新七種方案

首先我們來創建測試表

IF OBJECT_ID('Test') IS NOT NULL DROP TABLE TestCREATE TABLE Test( Id int, Name nchar(100), [Counter] int,primary key (Id), unique (Name));GO

解決方案一(開啟事務)

我們統一創建存儲過程通過來SQLQueryStress來測試并發情況,我們來看第一種情況。

IF OBJECT_ID('TestPro') IS NOT NULL DROP PROCEDURE TestPro;GO CREATE PROCEDURE TestPro ( @Id INT )AS DECLARE @Name NCHAR(100) = CAST(@Id AS NCHAR(100))  BEGIN TRANSACTION IF EXISTS ( SELECT 1    FROM Test    WHERE Id = @Id )  UPDATE Test  SET  [Counter] = [Counter] + 1  WHERE Id = @Id; ELSE  INSERT Test    ( Id, Name, [Counter] )  VALUES ( @Id, @Name, 1 ); COMMITGO

SQL,Server,并發處理,存在即更新

SQL,Server,并發處理,存在即更新

同時開啟100個線程和200個線程出現插入重復鍵的幾率比較少還是存在。

解決方案二(降低隔離級別為最低隔離級別UNCOMMITED)

IF OBJECT_ID('TestPro') IS NOT NULL DROP PROCEDURE TestPro;GO CREATE PROCEDURE TestPro ( @Id INT )AS DECLARE @Name NCHAR(100) = CAST(@Id AS NCHAR(100))  SET TRANSACTION ISOLATION LEVEL READ UNCOMMITTED BEGIN TRANSACTION IF EXISTS ( SELECT 1    FROM Test    WHERE Id = @Id )  UPDATE Test  SET  [Counter] = [Counter] + 1  WHERE Id = @Id; ELSE  INSERT Test    ( Id, Name, [Counter] )  VALUES ( @Id, @name, 1 ); COMMITGO

此時問題依舊和解決方案一無異(如果降低級別為最低隔離級別,如果行記錄為空,前一事務如果未進行提交,當前事務也能讀取到該行記錄為空,如果當前事務插入進去并進行提交,此時前一事務再進行提交此時就會出現插入重復鍵問題)

SQL,Server,并發處理,存在即更新

解決方案三(提升隔離級別為最高級別SERIALIZABLE)

IF OBJECT_ID('TestPro') IS NOT NULL DROP PROCEDURE TestPro;GO CREATE PROCEDURE TestPro ( @Id INT )AS DECLARE @Name NCHAR(100) = CAST(@Id AS NCHAR(100))  SET TRANSACTION ISOLATION LEVEL SERIALIZABLE BEGIN TRANSACTION IF EXISTS ( SELECT 1    FROM dbo.Test    WHERE Id = @Id )  UPDATE dbo.Test  SET  [Counter] = [Counter] + 1  WHERE Id = @Id; ELSE  INSERT dbo.Test    ( Id, Name, [Counter] )  VALUES ( @Id, @Name, 1 ); COMMITGO

在這種情況下更加糟糕,直接到會導致死鎖

SQL,Server,并發處理,存在即更新

此時將隔離級別提升為最高隔離級別會解決插入重復鍵問題,但是對于更新來獲取排它鎖而未提交,而此時另外一個進程進行查詢獲取共享鎖此時將造成進程間相互阻塞從而造成死鎖,所以從此知最高隔離級別有時候能夠解決并發問題但是也會帶來死鎖問題。

解決方案四(提升隔離級別+良好的鎖)

此時我們再來在添加最高隔離級別的基礎上增添更新鎖,如下:

IF OBJECT_ID('TestPro') IS NOT NULL DROP PROCEDURE TestPro;GO CREATE PROCEDURE TestPro ( @Id INT )AS DECLARE @Name NCHAR(100) = CAST(@Id AS NCHAR(100))  SET TRANSACTION ISOLATION LEVEL SERIALIZABLE BEGIN TRANSACTION IF EXISTS ( SELECT 1    FROM dbo.Test WITH(UPDLOCK)    WHERE Id = @Id )  UPDATE dbo.Test  SET  [Counter] = [Counter] + 1  WHERE Id = @Id; ELSE  INSERT dbo.Test    ( Id, Name, [Counter] )  VALUES ( @Id, @Name, 1 ); COMMITGO

SQL,Server,并發處理,存在即更新

運行多次均未發現出現什么異常,通過查詢數據時使用更新鎖而非共享鎖,這樣的話一來可以讀取數據但不阻塞其他事務,二來還確保自上次讀取數據后數據未被更改,這樣就解決了死鎖問題。貌似這樣的方案是可行得,如果是高并發不知是否可行。

解決方案五(提升隔離級別為行版本控制SNAPSHOT)

ALTER DATABASE UpsertTestDatabaseSET ALLOW_SNAPSHOT_ISOLATION ON ALTER DATABASE UpsertTestDatabaseSET READ_COMMITTED_SNAPSHOT ONGO IF OBJECT_ID('TestPro') IS NOT NULL DROP PROCEDURE TestPro;GO CREATE PROCEDURE TestPro ( @Id INT )AS DECLARE @Name NCHAR(100) = CAST(@Id AS NCHAR(100))  BEGIN TRANSACTION IF EXISTS ( SELECT 1    FROM dbo.Test    WHERE Id = @Id )  UPDATE dbo.Test  SET  [Counter] = [Counter] + 1  WHERE Id = @Id; ELSE  INSERT dbo.Test    ( Id, Name, [Counter] )  VALUES ( @Id, @Name, 1 ); COMMITGO

上述解決方案也會出現插入重復鍵問題不可取。

解決方案六(提升隔離級別+表變量)

IF OBJECT_ID('TestPro') IS NOT NULL DROP PROCEDURE TestPro;GO CREATE PROCEDURE TestPro ( @Id INT )AS DECLARE @Name NCHAR(100) = CAST(@Id AS NCHAR(100)) DECLARE @updated TABLE ( i INT );  SET TRANSACTION ISOLATION LEVEL SERIALIZABLE; BEGIN TRANSACTION UPDATE Test SET  [Counter] = [Counter] + 1 OUTPUT DELETED.Id   INTO @updated WHERE Id = @Id;  IF NOT EXISTS ( SELECT i     FROM @updated )  INSERT INTO Test    ( Id, Name, counter )  VALUES ( @Id, @Name, 1 ); COMMITGO

SQL,Server,并發處理,存在即更新

SQL,Server,并發處理,存在即更新

經過多次認證也是零錯誤,貌似通過表變量形式實現可行。

解決方案七(提升隔離級別+Merge)

通過Merge關鍵來實現存在即更新否則則插入,同時我們應該注意設置隔離級別為SERIALIZABLE否則會出現插入重復鍵問題,代碼如下:

IF OBJECT_ID('TestPro') IS NOT NULL DROP PROCEDURE TestPro;GO CREATE PROCEDURE TestPro ( @Id INT )AS DECLARE @Name NCHAR(100) = CAST(@Id AS NCHAR(100)) SET TRAN ISOLATION LEVEL SERIALIZABLE  BEGIN TRANSACTION MERGE Test AS [target] USING  ( SELECT @Id AS Id  ) AS source ON source.Id = [target].Id WHEN MATCHED THEN  UPDATE SET    [Counter] = [target].[Counter] + 1 WHEN NOT MATCHED THEN  INSERT ( Id, Name, [Counter] )  VALUES ( @Id, @Name, 1 ); COMMITGO

多次認證無論是并發100個線程還是并發200個線程依然沒有異常信息。

總結

本節我們詳細討論了在并發中如何處理存在即更新,否則即插入問題的解決方案,目前來講以上三種方案可行。

解決方案一(最高隔離級別 + 更新鎖)

IF OBJECT_ID('TestPro') IS NOT NULL DROP PROCEDURE TestPro;GO CREATE PROCEDURE TestPro ( @Id INT )AS DECLARE @Name NCHAR(100) = CAST(@Id AS NCHAR(100))  BEGIN TRANSACTION;  UPDATE dbo.Test WITH ( UPDLOCK, HOLDLOCK ) SET  [Counter] = [Counter] + 1 WHERE Id = @Id;  IF ( @@ROWCOUNT = 0 )  BEGIN   INSERT dbo.Test     ( Id, Name, [Counter] )   VALUES ( @Id, @Name, 1 );  END  COMMITGO

暫時只能想到這三種解決方案,個人比較推薦方案一和方案三, 請問您有何高見,請留下您的評論若可行,我將進行后續補充。

解決方案二(最高隔離級別 + 表變量)

IF OBJECT_ID('TestPro') IS NOT NULL DROP PROCEDURE TestPro;GO CREATE PROCEDURE TestPro ( @Id INT )AS DECLARE @Name NCHAR(100) = CAST(@Id AS NCHAR(100)) DECLARE @updated TABLE ( i INT );  SET TRANSACTION ISOLATION LEVEL SERIALIZABLE; BEGIN TRANSACTION UPDATE Test SET  [Counter] = [Counter] + 1 OUTPUT DELETED.id   INTO @updated WHERE id = @id;  IF NOT EXISTS ( SELECT i     FROM @updated )  INSERT INTO Test    ( Id, Name, counter )  VALUES ( @Id, @Name, 1 ); COMMITGO

解決方案三(最高隔離級別 + Merge)

IF OBJECT_ID('TestPro') IS NOT NULL DROP PROCEDURE TestPro;GO CREATE PROCEDURE TestPro ( @Id INT )AS DECLARE @Name NCHAR(100) = CAST(@Id AS NCHAR(100)) SET TRAN ISOLATION LEVEL SERIALIZABLE  BEGIN TRANSACTION MERGE Test AS [target] USING  ( SELECT @Id AS Id  ) AS source ON source.Id = [target].Id WHEN MATCHED THEN  UPDATE SET    [Counter] = [target].[Counter] + 1 WHEN NOT MATCHED THEN  INSERT ( Id, Name, [Counter] )  VALUES ( @Id, @Name, 1 ); COMMITGO

暫時只能想到這三種解決方案,個人比較推薦方案一和方案三, 請問您有何高見,請留下您的評論若可行,我將進行后續補充。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到MSSQL教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产极品jizzhd欧美| 久久成人国产精品| 久久躁日日躁aaaaxxxx| 国产欧美一区二区三区四区| 一区二区av在线| 国产91色在线免费| 日韩精品日韩在线观看| 国产精品色视频| 欧美电影免费观看高清完整| 在线观看亚洲区| 久久综合伊人77777尤物| 中文字幕在线看视频国产欧美在线看完整| 国内外成人免费激情在线视频| 亚洲色图18p| 青草青草久热精品视频在线网站| 伊人久久综合97精品| 成人免费看黄网站| 欧美电影《睫毛膏》| 日韩高清电影好看的电视剧电影| 日韩美女在线观看一区| 亚洲级视频在线观看免费1级| 久久91亚洲精品中文字幕| 91色视频在线观看| 日韩小视频在线| 亚洲成成品网站| 日韩电影中文字幕在线| 亚洲国产成人av在线| 亚洲精品美女视频| 国产精品美女久久久久久免费| 国产精品极品美女在线观看免费| 欧美精品videosex性欧美| 国产精品黄色av| 国产精品r级在线| 亚洲第一网站男人都懂| 国产精品福利网| 久久亚洲精品一区二区| 国产精品一区二区久久久| 日韩在线观看免费全集电视剧网站| 欧美日韩精品国产| 久久69精品久久久久久久电影好| 精品无人国产偷自产在线| 精品丝袜一区二区三区| 欧美人与性动交a欧美精品| 欧美日韩国产综合视频在线观看中文| 91天堂在线观看| 国产精品96久久久久久又黄又硬| 国产日韩亚洲欧美| 久久99精品久久久久久青青91| 日韩视频欧美视频| 国内伊人久久久久久网站视频| 国产美女久久精品| 日韩在线观看成人| 亚洲夜晚福利在线观看| 亚洲影院在线看| 亚洲人a成www在线影院| 国产亚洲精品一区二区| 欧美视频一区二区三区…| 欧美在线精品免播放器视频| 欧美在线一级va免费观看| 成人网中文字幕| 国内精品一区二区三区| 日本欧美国产在线| 久久香蕉频线观| 欧美精品福利在线| 日韩av日韩在线观看| 97色在线视频| 久久精视频免费在线久久完整在线看| 亚洲国产欧美在线成人app| 欧美大尺度电影在线观看| 91欧美日韩一区| 国产免费一区二区三区在线能观看| 久久久999精品免费| 亚洲第一精品夜夜躁人人爽| 久久精品色欧美aⅴ一区二区| 97在线观看视频| 日韩电影中文字幕在线| 亚洲毛茸茸少妇高潮呻吟| 欧美乱大交xxxxx另类电影| 日韩欧美在线观看视频| 久久精品国产69国产精品亚洲| 九色精品美女在线| 欧美另类老肥妇| 久久青草福利网站| 亚洲综合日韩中文字幕v在线| 欧美午夜视频一区二区| 毛片精品免费在线观看| 国产精品久久不能| 成人精品在线观看| 91久久久久久久久久久| 2019中文字幕在线免费观看| 91在线观看免费高清| 欧美人成在线视频| 亚洲精品二三区| 亚洲国产中文字幕久久网| 久久久久久久久久久成人| 国产成人jvid在线播放| 亚洲一区美女视频在线观看免费| 日韩在线视频观看正片免费网站| 欧美成人中文字幕| 岛国精品视频在线播放| 97精品国产aⅴ7777| 久久香蕉国产线看观看网| 日韩国产精品视频| 日韩男女性生活视频| 91精品国产自产在线| 成人综合国产精品| 中日韩美女免费视频网站在线观看| 日本一区二区不卡| 在线一区二区日韩| 久久久国产精品免费| 中文字幕精品—区二区| 日韩精品免费在线播放| 久久精品国产亚洲精品2020| 中文字幕亚洲在线| 91精品视频一区| 久久久国产精彩视频美女艺术照福利| 国产精品精品视频一区二区三区| 亚洲级视频在线观看免费1级| 不卡av电影院| 日韩av在线免费看| 国产欧美在线播放| 一本色道久久88精品综合| 日韩欧美精品免费在线| 国产在线精品播放| 欧美猛少妇色xxxxx| 欧美激情久久久久久| 国产日韩欧美中文| 色综合天天狠天天透天天伊人| 欧美在线影院在线视频| 亚洲第一网中文字幕| 中文字幕亚洲欧美日韩高清| 久久精品成人动漫| 国产精品久久久久91| 亚洲精品欧美一区二区三区| 久久成年人视频| 欧美大尺度激情区在线播放| 亚洲sss综合天堂久久| 欧美日韩国产综合视频在线观看中文| 欧美激情一区二区三区在线视频观看| 韩国三级电影久久久久久| 91网在线免费观看| 黑人与娇小精品av专区| 日韩中文字幕第一页| 97在线观看免费高清| 岛国av一区二区在线在线观看| 久久九九免费视频| 亚洲乱亚洲乱妇无码| 国产一区二区动漫| 91av国产在线| 国产乱人伦真实精品视频| 亚洲精品国产suv| 福利一区福利二区微拍刺激| 另类专区欧美制服同性| 激情成人在线视频| 日韩av在线免费| 日韩欧美国产视频| 亚洲精品网站在线播放gif| 日本久久中文字幕| 91精品国产免费久久久久久| 91亚洲国产成人久久精品网站| 国产精品国模在线| 欧美—级a级欧美特级ar全黄| 国产91免费观看| 久久久久久18|