鎖:咱們知道,最經常使用的處理多用戶併發訪問的方法是加鎖。當一個用戶鎖住數據庫中的某個對象時,其餘用戶就不能再訪問該對象。加鎖對併發訪問的影響體如今鎖的粒度上。好比,放在一個表上的鎖限制對整個表的併發訪問;放在數據頁上的鎖限制了對整個數據頁的訪問;放在行上的鎖只限制對該行的併發訪問。可見行鎖粒度最小,併發訪問最好,頁鎖粒度最大,表鎖介於2者之間。數據庫
鎖有兩種:悲觀鎖和樂觀鎖。悲觀鎖假定其餘用戶企圖訪問或者改變你正在訪問、更改的對象的機率是很高的,所以在悲觀鎖的環境中,在你開始改變此對象以前就將該對象鎖住,而且直到你提交了所做的更改以後才釋放鎖。悲觀的缺陷是不管是頁鎖仍是行鎖,加鎖的時間可能會很長,這樣可能會長時間的限制其餘用戶的訪問,也就是說悲觀鎖的併發訪問性很差。與悲觀鎖相反,樂觀鎖則認爲其餘用戶企圖改變你正在更改的對象的機率是很小的,所以樂觀鎖直到你準備提交所做的更改時纔將對象鎖住,當你讀取以及改變該對象時並不加鎖。可見樂觀鎖加鎖的時間要比悲觀鎖短,樂觀鎖能夠用較大的鎖粒度得到較好的併發訪問性能。可是若是第二個用戶剛好在第一個用戶提交更改以前讀取了該對象,那麼當他完成了本身的更改進行提交時,數據庫就會發現該對象已經變化了,這樣,第二個用戶不得不從新讀取該對象並做出更改。這說明在樂觀鎖環境中,會增長併發用戶讀取對象的次數。 併發
從數據庫廠商的角度看,使用樂觀的頁鎖是比較好的,尤爲在影響不少行的批量操做中能夠放比較少的鎖,從而下降對資源的需求提升數據庫的性能。再考慮彙集索引。在數據庫中記錄是按照彙集索引的物理順序存放的。若是使用頁鎖,當兩個用戶同時訪問更改位於同一數據頁上的相鄰兩行時,其中一個用戶必須等待另外一個用戶釋放鎖,這會明顯地下降系統的性能。interbase和大多數關係數據庫同樣,採用的是樂觀鎖,並且讀鎖是共享的,寫鎖是排他的。能夠在一個讀鎖上再放置讀鎖,但不能再放置寫鎖;你不能在寫鎖上再放置任何鎖。鎖是目前解決多用戶併發訪問的有效手段。性能
死鎖:當二或多個工做各自具備某個資源的鎖定,但其它工做嘗試要鎖定此資源,而形成工做永久封鎖彼此時,會發生死鎖。例如:spa
1. 事務 A 取得數據列 1 的共享鎖定。對象
2. 事務B 取得數據列 2 的共享鎖定。索引
3. 事務A 如今要求數據列 2 的獨佔鎖定,但會被封鎖直到事務B 完成並釋出對數據列 2 的共享鎖定爲止。事務
4. 事務B 如今要求數據列 1 的獨佔鎖定,但會被封鎖直到事務A 完成並釋出對數據列 1 的共享鎖定爲止。資源
等到事務B 完成後,事務A 才能完成,但事務B 被事務A 封鎖了。這個情況也稱爲「循環相依性」(Cyclic Dependency)。事務A 相依於事務B,而且事務B 也由於相依於事務A 而封閉了這個循環。it
例如如下操做就會產生死鎖,兩個鏈接互相阻塞對方的update。io
鏈接1:
begin tran
select * from customers
update customers set CompanyName = CompanyName
waitfor delay '00:00:05'
select * from Employees
–由於Employees被鏈接2鎖住了,因此這裏會阻塞。
update Employees set LastName = LastName
commit tran
鏈接2:
begin tran
select * from Employees
update Employees set LastName = LastName
waitfor delay '00:00:05'
select * from customers
--由於customers被鏈接1鎖住了,因此這裏會阻塞。
update customers set CompanyName = CompanyName
commit tran
SQL Server遇到死鎖時會自動殺死其中一個事務,而另外一個事務會正常結束(提交或回滾)。
SQL Server對殺死的鏈接返回錯誤代碼是1205,異常提示是:
Your transaction (process ID #52) was deadlocked on {lock | communication buffer | thRead} resources with another process and has been chosen as the deadlock victim. Rerun your transaction.