SQL Server--存在則更新問題

在博客園看到一篇討論特別多的文章「探討SQL Server併發處理存在就更新七種解決方案」,這種業務需求很常見:若是記錄存在就更新,不存在就插入。html

最多見的作法:併發

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 );
COMMIT

一個很明顯的問題,在高併發下可能存在操做同一條記錄的多個線程都進入到INSERT環節,致使插入失敗。app

上面問題緣由在於進入INSERT或UPDATE環節沒有「排他」鎖,若是每一個線程在進行插入或更新前就得到記錄的「排他鎖」,也就解決了其餘線程併發處理相同記錄的可能性。換個說法,一個蘿蔔一個坑,先無論這個坑有沒有蘿蔔,先把坑占上,再考慮其餘的。高併發

如何佔坑呢?並且是「排他地」佔坑呢?this

在SQL SERVER中,排他鎖即X鎖,對目標加X鎖有兩種方式:spa

一、使用SELECT+WITH(XLOCK)查詢提示線程

二、使用UPDATE/INSERT/DELETE操做code

雖然SELECT+WITH(XLOCK)查詢提示能作到加X鎖,可是這種X鎖有點「不靠譜」,MSDN給出解釋:htm

Using XLOCK in SELECT statements will not prevent reads from happening. This is because SQL Server has a special optimization under read committed isolation level that checks if the row is dirty or not and ignores the xlock if the row has not changed. Since this is acceptable under the read committed isolation level semantics it is by design. 

哪就只能UPDATE/INSERT/DELETE方式,DELETE確定排除,直接INSERT若是碰到記錄已存在又會報錯,最終只能選擇UPDATE,因而將業務需求實現爲:blog

BEGIN TRANSACTION
--先嚐試更新記錄佔坑
UPDATE  Test
SET     [Counter] = [Counter] + 1
WHERE   Id = @Id;

--若是更新操做沒有影響行,證實記錄不存在,則插入
IF @@ROWCOUNT<1
BEGIN
    INSERT  Test
            ( Id, Name, [Counter] )
    VALUES  ( @Id, @Name, 1 );
END
COMMIT

哪這樣真的把坑占上沒?

當ID=1記錄不存在時,執行下面SQL:

BEGIN TRANSACTION
--先嚐試更新記錄佔坑
UPDATE  Test
SET     [Counter] = [Counter] + 1
WHERE   Id = 1;

EXEC sp_lock @@SPID

雖然UPDATE的確會產生X排他鎖,可是沒有把鎖「持續」地佔下來,所以也沒法保證高併發下對該記錄的INSERT/UPDATE操做以「串行」方式執行。

要「持續」鎖,也有兩個辦法:

一、使用WITH(HOLDLOCK)鎖提示

二、使用SERIALIZABLE事務隔離級別

看下使用WITH(HOLDLOCK)鎖提示得到的鎖,一樣當ID=1記錄不存在時:

BEGIN TRANSACTION
--先嚐試更新記錄佔坑
UPDATE  Test WITH(HOLDLOCK)
SET     [Counter] = [Counter] + 1
WHERE   Id = 1;

EXEC sp_lock @@SPID

能夠看到除對錶和惟一索引上加IX和IS鎖以及頁上IX鎖外,還有一個KEY級別的範圍鎖RangX-X,因爲範圍X鎖的存在,任何其餘回話嘗試對此範圍的UPDATE和INSERT操做都將被阻塞,所以能夠繼續判斷是否須要插入。

固然,使用SERIALIZABLE隔離級別也是相同的效果

BEGIN TRANSACTION
SET TRANSACTION ISOLATION LEVEL SERIALIZABLE
--先嚐試更新記錄佔坑
UPDATE  Test
SET     [Counter] = [Counter] + 1
WHERE   Id = 1;

EXEC sp_lock @@SPID

======================================

更新後如何判斷是否須要插入呢?

方式1: 使用@@ROWCOUNT來判斷更新數據是否影響行,若是影響,則證實數據存在,無需更新

方式2: 使用IF NOT EXISTS也是能夠的,反正坑已經被占上,別的回話也不能改,再查一次就是多此一舉而已,不影響結果。

總的仍是推薦使用方式1,效率最高,避免一次SELECT操做,最後的腳本推薦爲:

BEGIN TRANSACTION
--先嚐試更新記錄佔坑
UPDATE  Test WITH(HOLDLOCK)
SET     [Counter] = [Counter] + 1
WHERE   Id = @Id;

--若是更新操做沒有影響行,證實記錄不存在,則插入
IF @@ROWCOUNT<1
BEGIN
    INSERT  Test
            ( Id, Name, [Counter] )
    VALUES  ( @Id, @Name, 1 );
END
COMMIT

================================================

PS1: 一般狀況下,我的不太推薦修改事務隔離級別,事務隔離級別影響的是整個事務,而鎖提示隻影響特定語句。

================================================

PS2: 上面業務查詢和更新都基於主鍵,在不少真實的業務場景下,主鍵一般爲非業務鍵即自增鍵,而須要根據業務鍵來操做,存在如下死鎖可能:

回話1: 先獲取彙集索引上X鎖,嘗試獲取非彙集索引上的X鎖

回話2:先獲取到非彙集索引上的X鎖,嘗試獲取彙集索引上X鎖

=================================================

語文很差,原本很簡單的一個東西,被本身描述成這樣,各位包含。

處理問題,先看原理,再考慮如何解決,才最簡單有效。

=================================================

相關文章
相關標籤/搜索