什麼是死鎖?
鎖,顧名思義,含義真的就是我們平常每天看到的那個鎖,鎖門的鎖,如果門鎖着,那就進不去了,那就隻能在門外等着。
軟體中的鎖,意義和這個類似,也是為了阻止非授權使用者能夠進入某些代碼的執行,如果要想執行被鎖保護(同步)的代碼,那麼必須要先獲得鎖,如果你想進去的時候,别人正在擁有這個鎖,你也隻好等待了,必須等到被人使用完了以後才能進入被保護或者被同步的代碼執行。
什麼是死鎖呢?
死鎖是指兩個或兩個以上的線程或者程序在執行過程中,由于競争資源或者由于彼此通信而造成的一種阻塞的現象,而且會一直阻塞下去,此時稱軟體或者系統處于死鎖狀态。
為什麼會死鎖?
我們來看一張圖
這裡有兩個線程,線程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