什麼是死鎖?
鎖,顧名思義,含義真的就是我們平常每天看到的那個鎖,鎖門的鎖,如果門鎖着,那就進不去了,那就只能在門外等着。
軟件中的鎖,意義和這個類似,也是爲了阻止非授權用戶能夠進入某些代碼的執行,如果要想執行被鎖保護(同步)的代碼,那麼必須要先獲得鎖,如果你想進去的時候,別人正在擁有這個鎖,你也只好等待了,必須等到被人使用完了以後才能進入被保護或者被同步的代碼執行。
什麼是死鎖呢?
死鎖是指兩個或兩個以上的線程或者進程在執行過程中,由於競爭資源或者由於彼此通信而造成的一種阻塞的現象,而且會一直阻塞下去,此時稱軟件或者系統處於死鎖狀態。
爲什麼會死鎖?
我們來看一張圖
這裏有兩個線程,線程A和線程B,這裏也有兩把鎖,鎖1和鎖2,線程A和線程B分別按照自己的方式來使用這兩把鎖,線程1先獲取鎖1,做一些事情,然後獲取鎖2,再做一些事情,最後釋放鎖2,釋放鎖1,線程B先獲取鎖2,做一些事情,然後獲取鎖1,再做一些事情,最後釋放鎖1,釋放鎖2.
看起來沒有問題,而且往往執行起來也沒有什麼問題,但是,恰恰就是這樣的使用鎖,就是導致死鎖的原因之一。
我們假設線程A獲得鎖1的同時線程B也獲得了鎖2,一旦這種情況發生,就發生了死鎖,爲什麼呢?
因爲線程A接下來要去獲取鎖2,因爲線程B還沒有釋放鎖2,所以線程A只好等待,同樣線程B做了一些事情之後,需要再去獲取鎖1,同樣的原因,線程A還沒有釋放鎖!,因爲還在等待鎖2呢,所以,線程A和線程B就進入了死鎖狀態。會一直等待下去,誰也得不到自己想要的鎖。
如何解決死鎖呢?
要解決死鎖問題,必須要正確使用鎖,可以從兩個方面來解決死鎖問題:
1.一個線程使用多個鎖的時候,使用順序保持一致
2.一個線程使用多個鎖的時候,鎖儘量不要交叉,一個鎖用完,立即釋放後再去使用另外一個鎖。
只要注意了這兩點,基本上就不會死鎖了,當然如果使用不當,比如循環用鎖等,遞歸使用等,也是有可能發生死鎖的。
如何調試死鎖?
VC和Windbg都提供了很好的調試死鎖的方法,這裏有一個視頻是關於VC如何調試死鎖問題的,如果有興趣,可以看一下。
https://edu.csdn.net/course/detail/28915