python lock_python多线程Lock和RLock的区别
python多線程Lock和RLock的區(qū)別
1. 兩種鎖的不同
1.1 定義
為了確保對共享資源的訪問,python提供了兩種鎖,一個是上一篇提到的Lock,還有一個就是RLock,他們的區(qū)別在于:
Lock是可用的最低級別的同步指令,一個線程只能請求一次,而RLock是可以被一個線程請求多次的同步指令
當(dāng)Lock處于鎖定狀態(tài)時,不被特定的線程所擁有,而RLock使用了“擁有的線程”和“遞歸等級”的概念,因此處于鎖定狀態(tài)時,可以被線程擁有
1.2 死鎖
Lock在下面的情形下會發(fā)生死鎖
Lock.acquire()
Lock.acquire()
Lock.release()
Lock.release()
連續(xù)兩次acquire請求,會導(dǎo)致死鎖,因為第一次獲得鎖之后還沒有釋放時,第二次acquire請求緊接著就到來,可是acquire會讓程序阻塞,無法執(zhí)行release(),這就導(dǎo)致鎖永遠(yuǎn)無法釋放,死鎖是非常危險非常嚴(yán)重的問題
1.3 可重入鎖
RLock就不存在1.2中所提到的死鎖問題
RLock.acquire()
RLock.acquire()
RLock.release()
RLock.release()
不過要保證有多少次acquire(),就有多少次release()
2. 怎么會多次請求鎖呢?
最初接觸到Lock和RLock這兩者之間的不同之處時,感到十分困惑。RLock的優(yōu)勢在于,在同一個線程里可以多次申請鎖,而Lock則不能,必須在釋放之后才能再次申請,那么,這樣做也沒問題啊,不會出現(xiàn)第一次申請后,在釋放前又申請的可能啊,在編寫代碼的時候,完全可以認(rèn)為的控制這種情況的發(fā)生。
然而事實并非如此,我現(xiàn)在假設(shè)一種情形,使得死鎖的發(fā)生不可避免
import threading
m_lock = threading.Lock()
def h():
with m_lock:
print('h')
def g():
with m_lock:
print('g')
h()
g()
上面的例子中,h()和g()中都用了Lock,在多線程環(huán)境下,他們可以做到相安無事,但是,程序的結(jié)構(gòu)總是處于變化中,尤其是那些龐大的系統(tǒng),一個小小的變化可能牽一發(fā)而動全身,假設(shè)發(fā)生了下面的變化
import threading
m_lock = threading.Lock()
# m_lock = threading.RLock()
def h():
with m_lock:
g()
print('h')
def g():
with m_lock:
print('g')
h()
g()
在h()函數(shù)中,獲得鎖以后要執(zhí)行g(shù)(),那么此時,程序就會發(fā)生死鎖,在大的項目里,情況會比這更加復(fù)雜,你很難通過眼前的幾行代碼發(fā)現(xiàn)這種死鎖的情況,因為很可能發(fā)生死鎖的地方是在很深層次的調(diào)用過程中,因此,使用RLock是非常安全的選擇.
執(zhí)行上面的代碼,程序不會輸出任何信息,也永遠(yuǎn)不會結(jié)束,因為已經(jīng)發(fā)生了死鎖,將注釋的Lock替換成RLock,程序立馬可以執(zhí)行
總結(jié)
以上是生活随笔為你收集整理的python lock_python多线程Lock和RLock的区别的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 国债三年期利息是多少?
- 下一篇: python 的csr_python的高