您好,登錄后才能下訂單哦!
1、GIL簡介
GIL的全稱為Global Interpreter Lock,全局解釋器鎖。
1.1 GIL設(shè)計(jì)理念與限制
python的代碼執(zhí)行由python虛擬機(jī)(也叫解釋器主循環(huán),CPython版本)來控制,python在設(shè)計(jì)之初就考慮到在解釋器的主循環(huán)中,同時(shí)只有一個(gè)線程在運(yùn)行。即在任意時(shí)刻只有一個(gè)線程在解釋器中運(yùn)行。對python虛擬機(jī)訪問的控制由全局解釋鎖GIL控制,正是這個(gè)鎖來控制同一時(shí)刻只有一個(gè)線程能夠運(yùn)行。
在調(diào)用外部代碼(如C、C++擴(kuò)展函數(shù))的時(shí)候,GIL將會(huì)被鎖定,直到這個(gè)函數(shù)結(jié)束為止(由于期間沒有python的字節(jié)碼運(yùn)行,所以不會(huì)做線程切換)。
在python中使用都是操作系統(tǒng)級別的線程,linux中使用的pthread,window使用的是其原生線程。
從上面的概述中可以直觀的看出py在同一時(shí)刻只能跑一個(gè)線程,這樣在跑多線程的情況下,只有當(dāng)線程獲取到全局解釋器鎖后才能運(yùn)行,而全局解釋器鎖只有一個(gè),因此即使在多核的情況下也只能發(fā)揮出單核的功能。
那么這樣看起來py不給力啊,GIL直接導(dǎo)致CPython不能利用物理多核的性能加速運(yùn)行。那么為什么會(huì)有這樣的設(shè)計(jì)?考慮到Guido van Rossum 在創(chuàng)造python的時(shí)候,上世紀(jì)90年代,多核cpu完全屬于不可想象的,現(xiàn)在由于硬件發(fā)展速度太快,程序編寫就要考慮用盡cpu的全部性能,否則就要被淘汰,那么對于python同樣也要如此。
上面主要說的是這種設(shè)計(jì)的劣勢,下面再討論它的優(yōu)勢。
GIL的設(shè)計(jì)簡化了CPython的實(shí)現(xiàn),使得對象模型,包括關(guān)鍵的內(nèi)建類型如字典,都隱式可以并發(fā)訪問。鎖住全局解釋器使得其比較容易的實(shí)現(xiàn)對多線程的支持,但也折損了多處理器主機(jī)的并行計(jì)算能力。
但是不論標(biāo)準(zhǔn)的,還是第三方的擴(kuò)展模塊,都被設(shè)計(jì)成在進(jìn)行密集計(jì)算任務(wù)時(shí)釋放GIL。另外還有在做IO操作時(shí),GIL總是被釋放。對所有面對內(nèi)建的操作系統(tǒng)C代碼的程序來說,GIL會(huì)在這個(gè)IO調(diào)用之前被釋放,以允許其它的線程在等待這個(gè)IO的時(shí)候運(yùn)行。如果是純計(jì)算的程序,沒有IO操作,解釋器會(huì)每隔100次或每隔一定時(shí)間15ms去釋放GIL。
這里可以理解為IO密集型的python比計(jì)算密集型的程序更能利用多線程環(huán)境帶來的便利。
1.2 GIL對線程執(zhí)行的影響
多線程環(huán)境中,python虛擬機(jī)按照以下方式執(zhí)行:
上節(jié)說到python語言和程序一樣要考慮用盡cpu的性能,下面在討論py的應(yīng)對方法。
python的應(yīng)對方法很簡單,在新的python3中依然有GIL,原因大概有下幾點(diǎn):
然后最重要的還在于以下幾個(gè)方面,簡單來說就是py不改,一樣能實(shí)現(xiàn)我們的需求。
python中GIL使得同一個(gè)時(shí)刻只有一個(gè)線程在一個(gè)cpu上執(zhí)行,無法將多個(gè)線程映射到多個(gè)cpu上執(zhí)行,但GIL并不會(huì)一直占有,它會(huì)在適當(dāng)?shù)臅r(shí)候釋放
import threading count = 0 def add(): global count for i in range(10**6): count += 1 def minus(): global count for i in range(10**6): count -= 1 thread1 = threading.Thread(target=add) thread2 = threading.Thread(target=minus) thread1.start() thread2.start() thread1.join() thread2.join() print(count)
分別運(yùn)行三次的結(jié)果:
-59452
60868
-77007
可以看到count并不是一個(gè)固定值,說明GIL會(huì)在某個(gè)時(shí)刻釋放,那么GIL具體在什么情況下釋放呢:
1.執(zhí)行的字節(jié)碼行數(shù)到達(dá)一定閾值
2.通過時(shí)間片劃分,到達(dá)一定時(shí)間閾值
3.在遇到IO操作時(shí),主動(dòng)釋放
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對億速云的支持。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。