淺析SQL Server的鎖機制

導讀: 各類大型數據庫所採用的鎖的基本理論是一致的,但在具體實現上各有差異。SQL Server更強調由系統來管理鎖。在用戶有SQL請求時,系統分析請求,自動在知足鎖定條件和系統性能之間爲數據庫加上適當的鎖,同時系統在運行期間經常自動進行優化處理,實行動態加鎖。對於通常的用戶而言,經過系統的自動鎖定管理機制基本能夠知足使用要求,但若是對數據安全、數據庫完整性和一致性有特殊要求,就須要瞭解SQL Server的鎖機制,掌握數據庫鎖定方法。]  
    鎖是數據庫中的一個很是重要的概念,它主要用於多用戶環境下保證數據庫完整性和一致性。 咱們知道,多個用戶可以同時操縱同一個數據庫中的數據,會發生數據不一致現象。即若是沒有鎖定且多個用戶同時訪問一個數據庫,則當他們的事務同時使用相同的數據時可能會發生問題。這些問題包括:丟失更新、髒讀、不可重複讀和幻覺讀:
1.當兩個或多個事務選擇同一行,而後基於最初選定的值更新該行時,會發生丟失更新問題。每一個事務都不知道其它事務的存在。最後的更新將重寫由其它事務所作的更新,這將致使數據丟失。例如,兩個編輯人員製做了同一文檔的電子複本。每一個編輯人員獨立地更改其複本,而後保存更改後的複本,這樣就覆蓋了原始文檔。最後保存其更改複本的編輯人員覆蓋了第一個編輯人員所作的更改。若是在第一個編輯人員完成以後第二個編輯人員才能進行更改,則能夠避免該問題。
2. 髒讀就是指當一個事務正在訪問數據,而且對數據進行了修改,而這種修改尚未提交到數據庫中,這時,另一個事務也訪問這個數據,而後使用了這個數據。由於這個數據是尚未提交的數據,那麼另一個事務讀到的這個數據是髒數據,依據髒數據所作的操做多是不正確的。例如,一個編輯人員正在更改電子文檔。在更改過程當中,另外一個編輯人員複製了該文檔(該複本包含到目前爲止所作的所有更改)並將其分發給預期的用戶。此後,第一個編輯人員認爲目前所作的更改是錯誤的,因而刪除了所作的編輯並保存了文檔。分發給用戶的文檔包含再也不存在的編輯內容,而且這些編輯內容應認爲從未存在過。若是在第一個編輯人員肯定最終更改前任何人都不能讀取更改的文檔,則能夠避免該問題。
3.不可重複讀是指在一個事務內,屢次讀同一數據。在這個事務尚未結束時,另一個事務也訪問該同一數據。那麼,在第一個事務中的兩次讀數據之間,因爲第二個事務的修改,那麼第一個事務兩次讀到的的數據多是不同的。這樣就發生了在一個事務內兩次讀到的數據是不同的,所以稱爲是不可重複讀。例如,一個編輯人員兩次讀取同一文檔,但在兩次讀取之間,做者重寫了該文檔。當編輯人員第二次讀取文檔時,文檔已更改。原始讀取不可重複。若是隻有在做者所有完成編寫後編輯人員才能夠讀取文檔,則能夠避免該問題。
4.幻覺讀是指當事務不是獨立執行時發生的一種現象,例如第一個事務對一個表中的數據進行了修改,這種修改涉及到表中的所有數據行。同時,第二個事務也修改這個表中的數據,這種修改是向表中插入一行新數據。那麼,之後就會發生操做第一個事務的用戶發現表中還有沒有修改的數據行,就好象發生了幻覺同樣。例如,一個編輯人員更改做者提交的文檔,但當生產部門將其更改內容合併到該文檔的主複本時,發現做者已將未編輯的新材料添加到該文檔中。若是在編輯人員和生產部門完成對原始文檔的處理以前,任何人都不能將新材料添加到文檔中,則能夠避免該問題。
因此,處理多用戶併發訪問的方法是加鎖。鎖是防止其餘事務訪問指定的資源控制、實現併發控制的一種主要手段。當一個用戶鎖住數據庫中的某個對象時,其餘用戶就不能再訪問該對象。加鎖對併發訪問的影響體如今鎖的粒度上。爲了控制鎖定的資源,應該首先了解系統的空間管理。在SQL Server 2000系統中,最小的空間管理單位是頁,一個頁有8K。全部的數據、日誌、索引都存放在頁上。另外,使用頁有一個限制,這就是表中的一行數據必須在同一個頁上,不能跨頁。頁上面的空間管理單位是盤區,一個盤區是8個連續的頁。表和索引的最小佔用單位是盤區。數據庫是由一個或者多個表或者索引組成,便是由多個盤區組成。放在一個表上的鎖限制對整個表的併發訪問;放在盤區上的鎖限制了對整個盤區的訪問;放在數據頁上的鎖限制了對整個數據頁的訪問;放在行上的鎖只限制對該行的併發訪問。
SQL Server 2000 具備多粒度鎖定,容許一個事務鎖定不一樣類型的的資源。爲了使鎖定的成本減至最少,SQL Server 自動將資源鎖定在適合任務的級別。鎖定在較小的粒度(例如行)能夠增長併發但須要較大的開銷,由於若是鎖定了許多行,則須要控制更多的鎖。鎖定在較大的粒度(例如表)就併發而言是至關昂貴的,由於鎖定整個表限制了其它事務對錶中任意部分進行訪問,但要求的開銷較低,由於須要維護的鎖較少。SQL Server 能夠鎖定行、頁、擴展盤區、表、庫等資源。
行是能夠鎖定的最小空間, 行級鎖佔用的數據資源最少,因此在事務的處理過程當中,容許其餘事務繼續操縱同一個表或者同一個頁的其餘數據,大大下降了其餘事務等待處理的時間,提升了系統的併發性。
頁級鎖是指在事務的操縱過程當中,不管事務處理數據的多少,每一次都鎖定一頁,在這個頁上的數據不能被其餘事務操縱。在SQL Server 7.0之前,使用的是頁級鎖。頁級鎖鎖定的資源比行級鎖鎖定的數據資源多。在頁級鎖中,即便是一個事務只操縱頁上的一行數據,那麼該頁上的其餘數據行也不能被其餘事務使用。所以,當使用頁級鎖時,會出現數據的浪費現象,也就是說,在同一個頁上會出現數據被佔用卻沒有使用的現象。在這種現象中,數據的浪費最多不超過一個頁上的數據行。
表級鎖也是一個很是重要的鎖。表級鎖是指事務在操縱某一個表的數據時,鎖定了這個數據所在的整個表,其餘事務不能訪問該表中的其餘數據。當事務處理的數據量比較大時,通常使用表級鎖。表級鎖的特色是使用比較少的系統資源,可是卻佔用比較多的數據資源。與行級鎖和頁級鎖相比,表級鎖佔用的系統資源例如內存比較少,可是佔用的數據資源倒是最大。在表級鎖時,有可能出現數據的大量浪費現象,由於表級鎖鎖定整個表,那麼其餘的事務都不能操縱表中的其餘數據。
盤區鎖是一種特殊類型的鎖,只能用在一些特殊的狀況下。簇級鎖就是指事務佔用一個盤區,這個盤區不能同時被其餘事務佔用。例如在建立數據庫和建立表時,系統分配物理空間時使用這種類型的鎖。系統是按照盤區分配空間的。當系統分配空間時,使用盤區鎖,防止其餘事務同時使用同一個盤區。當系統完成分配空間以後,就再也不使用這種類型的盤區鎖。特別是,當涉及到對數據操做的事務時,不使用盤區鎖。
數據庫級鎖是指鎖定整個數據庫,防止任何用戶或者事務對鎖定的數據庫進行訪問。數據庫級鎖是一種很是特殊的鎖,它只是用於數據庫的恢復操做過程當中。這種等級的鎖是一種最高等級的鎖,由於它控制整個數據庫的操做。只要對數據庫進行恢復操做,那麼就須要設置數據庫爲單用戶模式,這樣系統就能防止其餘用戶對該數據庫進行各類操做。
行級鎖是一種最優鎖,由於行級鎖不可能出現數據既被佔用又沒有使用的浪費現象。可是,若是用戶事務中頻繁對某個表中的多條記錄操做,將致使對該表的許多記錄行都加上了行級鎖,數據庫系統中鎖的數目會急劇增長,這樣就加劇了系統負荷,影響系統性能。所以,在SQL Server中,還支持鎖升級(lock escalation)。所謂鎖升級是指調整鎖的粒度,將多個低粒度的鎖替換成少數的更高粒度的鎖,以此來下降系統負荷。在SQL Server中當一個事務中的鎖較多,達到鎖升級門限時,系統自動將行級鎖和頁面鎖升級爲表級鎖。特別值得注意的是,在SQL Server中,鎖的升級門限以及鎖升級是由系統自動來肯定的,不須要用戶設置。
       在SQL Server數據庫中加鎖時,除了能夠對不一樣的資源加鎖,還可使用不一樣程度的加鎖方式,即鎖有多種模式,SQL Server中鎖模式包括:    
1.共享鎖     SQL Server中,共享鎖用於全部的只讀數據操做。共享鎖是非獨佔的,容許多個併發事務讀取其鎖定的資源。默認狀況下,數據被讀取後,SQL Server當即釋放共享鎖。例如,執行查詢「SELECT * FROM AUTHORS」時,首先鎖定第一頁,讀取以後,釋放對第一頁的鎖定,而後鎖定第二頁。這樣,就容許在讀操做過程當中,修改未被鎖定的第一頁。可是,事務隔離級別鏈接選項設置和SELECT語句中的鎖定設置均可以改變SQL Server的這種默認設置。例如,「 SELECT * FROM AUTHORS HOLDLOCK」就要求在整個查詢過程當中,保持對錶的鎖定,直到查詢完成才釋放鎖定。   
2.更新鎖     更新鎖在修改操做的初始化階段用來鎖定可能要被修改的資源,這樣能夠避免使用共享鎖形成的死鎖現象。由於使用共享鎖時,修改數據的操做分爲兩步,首先得到一個共享鎖,讀取數據,而後將共享鎖升級爲排它鎖,而後再執行修改操做。這樣若是同時有兩個或多個事務同時對一個事務申請了共享鎖,在修改數據的時候,這些事務都要將共享鎖升級爲排它鎖。這時,這些事務都不會釋放共享鎖而是一直等待對方釋放,這樣就形成了死鎖。若是一個數據在修改前直接申請更新鎖,在數據修改的時候再升級爲排它鎖,就能夠避免死鎖。
3.排它鎖     排它鎖是爲修改數據而保留的。它所鎖定的資源,其餘事務不能讀取也不能修改。  
4.結構鎖     執行表的數據定義語言 (DDL) 操做(例如添加列或除去表)時使用架構修改 (Sch-M) 鎖。當編譯查詢時,使用架構穩定性 (Sch-S) 鎖。架構穩定性 (Sch-S) 鎖不阻塞任何事務鎖,包括排它鎖。所以在編譯查詢時,其它事務(包括在表上有排它鎖的事務)都能繼續運行。但不能在表上執行 DDL 操做。
5.意向鎖     意向鎖說明SQL Server有在資源的低層得到共享鎖或排它鎖的意向。例如,表級的共享意向鎖說明事務意圖將排它鎖釋放到表中的頁或者行。意向鎖又能夠分爲共享意向鎖、獨佔意向鎖和共享式獨佔意向鎖。共享意向鎖說明事務意圖在共享意向鎖所鎖定的低層資源上放置共享鎖來讀取數據。獨佔意向鎖說明事務意圖在共享意向鎖所鎖定的低層資源上放置排它鎖來修改數據。共享式排它鎖說明事務容許其餘事務使用共享鎖來讀取頂層資源,並意圖在該資源低層上放置排它鎖。    
6.大容量更新鎖 當將數據大容量複製到表,且指定了 TABLOCK 提示或者使用 sp_tableoption 設置了 table lock on bulk 表選項時,將使用大容量更新 鎖。大容量更新鎖容許進程將數據併發地大容量複製到同一表,同時防止其它不進行大容量複製數據的進程訪問該表。
   
SQL Server系統中建議讓系統自動管理鎖,該系統會分析用戶的SQL語句要求,自動爲該請求加上合適的鎖,並且在鎖的數目太多時,系統會自動進行鎖升級。如前所述,升級的門限由系統自動配置,並不須要用戶配置。 在實際應用中,有時爲了應用程序正確運行和保持數據的一致性,必須人爲地給數據庫的某個表加鎖。好比,在某應用程序的一個事務操做中,須要根據一編號對幾個數據表作統計操做,爲保證統計數據時間的一致性和正確性,從統計第一個表開始到所有表結束,其餘應用程序或事務不能再對這幾個表寫入數據,這個時候,該應用程序但願在從統計第一個數據表開始或在整個事務開始時可以由程序人爲地(顯式地)鎖定這幾個表,這就須要用到手工加鎖(也稱顯式加鎖)技術。
可使用 SELECT、INSERT、UPDATE 和 DELETE 語句指定表級鎖定提示的範圍,以引導 Microsoft SQL Server 2000 使用所需的鎖類型。當須要對對象所得到鎖類型進行更精細控制時,使用表級鎖定提示更改默認的鎖定行爲。
所指定的表級鎖定提示有以下幾種:   
1. HOLDLOCK: 在該表上保持共享鎖,直到整個事務結束,而不是在語句執行完當即釋放所添加的鎖。    
2. NOLOCK:不添加共享鎖和排它鎖,當這個選項生效後,可能讀到未提交讀的數據或「髒數據」,這個選項僅僅應用於SELECT語句。    
3. PAGLOCK:指定添加頁鎖(不然一般可能添加表鎖)。  
4. READCOMMITTED用與運行在提交讀隔離級別的事務相同的鎖語義執行掃描。默認狀況下,SQL Server 2000 在此隔離級別上操做。。  
5. READPAST: 跳過已經加鎖的數據行,這個選項將使事務讀取數據時跳過那些已經被其餘事務鎖定的數據行,而不是阻塞直到其餘事務釋放鎖,READPAST僅僅應用於READ COMMITTED隔離性級別下事務操做中的SELECT語句操做。   
6. READUNCOMMITTED:等同於NOLOCK。    
7. REPEATABLEREAD:設置事務爲可重複讀隔離性級別。  
8. ROWLOCK:使用行級鎖,而不使用粒度更粗的頁級鎖和表級鎖。   
9. SERIALIZABLE:用與運行在可串行讀隔離級別的事務相同的鎖語義執行掃描。等同於 HOLDLOCK。
10. TABLOCK:指定使用表級鎖,而不是使用行級或頁面級的鎖,SQL Server在該語句執行完後釋放這個鎖,而若是同時指定了HOLDLOCK,該鎖一直保持到這個事務結束。    
11. TABLOCKX:指定在表上使用排它鎖,這個鎖能夠阻止其餘事務讀或更新這個表的數據,直到這個語句或整個事務結束。  
12. UPDLOCK :指定在讀表中數據時設置更新 鎖(update lock)而不是設置共享鎖,該鎖一直保持到這個語句或整個事務結束,使用UPDLOCK的做用是容許用戶先讀取數據(並且不阻塞其餘用戶讀數據),而且保證在後來再更新數據時,這一段時間內這些數據沒有被其餘用戶修改。   
死鎖問題
在數據庫系統中,死鎖是指多個用戶(進程)分別鎖定了一個資源,並又試圖請求鎖定對方已經鎖定的資源,這就產生了一個鎖定請求環,致使多個用戶(進程)都處於等待對方釋放所鎖定資源的狀態。這種死鎖是最典型的死鎖形式, 例如在同一時間內有兩個事務A和B,事務A有兩個操做:鎖定表part和請求訪問表supplier;事務B也有兩個操做:鎖定表supplier和請求訪問表part。結果,事務A和事務B之間發生了死鎖。
    死鎖的第二種狀況是,當在一個數據庫中時,有若干個長時間運行的事務執行並行的操做,當查詢分析器處理一種很是複雜的查詢例如鏈接查詢時,那麼因爲不能控制處理的順序,有可能發生死鎖現象。
    在SQL Server中,系統可以自動按期搜索和處理死鎖問題。系統在每次搜索中標識全部等待鎖定請求的進程會話,若是在下一次搜索中該被標識的進程仍處於等待狀態,SQL Server就開始遞歸死鎖搜索。當搜索檢測到鎖定請求環時,SQL Server 經過自動選擇能夠打破死鎖的線程(死鎖犧牲品)來結束死鎖。SQL Server 回滾做爲死鎖犧牲品的事務,通知線程的應用程序(經過返回 1205 號錯誤信息),取消線程的當前請求,而後容許不間斷線程的事務繼續進行。SQL Server 一般選擇運行撤消時花費最少的事務的線程做爲死鎖犧牲品。另外,用戶可使用 SET 語句將會話的 DEADLOCK_PRIORITY 設置爲 LOW。DEADLOCK_PRIORITY 選項控制在死鎖狀況下如何衡量會話的重要性。若是會話的設置爲 LOW ,則當會話陷入死鎖狀況時將成爲首選犧牲品。
    理解了死鎖的概念,在應用程序中就能夠採用下面的一些方法來儘可能避免死鎖了: (1)合理安排表訪問順序。 (2)在事務中儘可能避免用戶干預,儘可能使一個事務處理的任務少些, 保持事務簡短並在一個批處理中。 (3)數據訪問時域離散法, 數據訪問時域離散法是指在客戶機/服務器結構中,採起各類控制手段控制對數據庫或數據庫中的對象訪問時間段。主要經過如下方式實現: 合理安排後臺事務的執行時間,採用工做流對後臺事務進行統一管理。工做流在管理任務時,一方面限制同一類任務的線程數(每每限制爲1個),防止資源過多佔用; 另外一方面合理安排不一樣任務執行時序、時間,儘可能避免多個後臺任務同時執行,另外, 避免在前臺交易高峯時間運行後臺任務。 (4)數據存儲空間離散法。數據存儲空間離散法是指採起各類手段,將邏輯上在一個表中的數據分散到若干離散的空間上去,以便改善對錶的訪問性能。主要經過如下方法實現: 第一,將大表按行或列分解爲若干小表; 第二,按不一樣的用戶羣分解。 (5)使用盡量低的隔離性級別。隔離性級別是指爲保證數據庫數據的完整性和一致性而使多用戶事務隔離的程度,SQL92定義了4種隔離性級別:未提交讀、提交讀、可重複讀和可串行。若是選擇太高的隔離性級別,如可串行,雖然系統能夠因實現更好隔離性而更大程度上保證數據的完整性和一致性,但各事務間衝突而死鎖的機會大大增長,大大影響了系統性能。 (6)使用綁定鏈接, 綁定鏈接容許兩個或多個事務鏈接共享事務和鎖,並且任何一個事務鏈接要申請鎖如同另一個事務要申請鎖同樣,所以能夠容許這些事務共享數據而不會有加鎖的衝突。   
    總之,瞭解SQL Server的鎖機制,掌握數據庫鎖定方法, 對一個合格的DBA來講是很重要的。
相關文章
相關標籤/搜索