新聞中心
Python子線程在創(chuàng)建自身的線程狀態(tài)對象后,會通過_PyGILState_NoteThreadState這個語句將這個對象放入到線程狀態(tài)對象鏈表中,當前活動的Python子線程不一定是獲得了GIL的線程。

洪雅網站制作公司哪家好,找創(chuàng)新互聯(lián)!從網頁設計、網站建設、微信開發(fā)、APP開發(fā)、自適應網站建設等網站項目制作,到程序開發(fā),運營維護。創(chuàng)新互聯(lián)2013年開創(chuàng)至今到現在10年的時間,我們擁有了豐富的建站經驗和運維經驗,來保證我們的工作的順利進行。專注于網站建設就選創(chuàng)新互聯(lián)。
在thread1.py中主線程現在是獲得了GIL的,但是子線程到現在還沒有申請GIL,自然也不會將自身掛起。由于主線程和子線程都是Win32的原生線程。所以操作系統(tǒng)可能在主線程和Python子線程之間切換。我們在這里要著重指出操作系統(tǒng)級的線程調度和Python級的線程調度是不同的。
Python級的線程調度一定意味著GIL擁有權的易手,而操作系統(tǒng)級的線程調度并不一定意味著GIL的易手,當所有的線程都完成了初始化動作之后。操作系統(tǒng)的線程調度和Python的線程調度才會同一。那時,Python的線程調度會迫使當前活動線程釋放GIL,而這一操作會觸發(fā)GIL中維護的Event內核對象。
這個觸發(fā)又進而觸發(fā)操作系統(tǒng)的線程調度。而在線程的初始化完成之前,在Python線程調度和操作系統(tǒng)線程調度之間并沒有這樣的因果關系。顯示了GIL在Python級線程調度與操作系統(tǒng)級線程調度之間所起的橋梁作用。
前面我們已經剖析過PyEval_AcquireThread的代碼,在PyEval_AcquireThread中,子線程進行了***的沖刺,它要生存,要執(zhí)行,于是它開始通過PyThread_acquire_ lock爭取GIL。到了這一步。
Python子線程將自己掛起,操作系統(tǒng)的線程調度機制再也不能靠自身的力量將其喚醒,只有等待Python的線程調度機制強迫主線程放棄GIL后。子線程才會被喚醒;而子線程被喚醒之后,主線程卻又陷入了苦苦地等待中,同樣苦苦地守望著Python強迫子線程放棄GIL的那一刻。
當子線程被Python的線程調度機制喚醒之后,它所作的***件事就是通過PyThreadState_Swap將Python維護的當前線程狀態(tài)對象設置為其自身的狀態(tài)對象,一如操作系統(tǒng)的進程上下文環(huán)境恢復一樣。
現在我們的Python子線程開始等待GIL,但是注意,線程的初始化還沒有真正完成,因為子線程還沒有順利進入字節(jié)碼解釋器。當Python線程調度將子線程喚醒之后。子線程將回到t_bootstrap中。
并進入PyEval_CallObjectWithKeywords,從這里一直往前,最終將調用PyEval_EvalFrameEx,進入解釋器。到了那個時候,Python子線程和主線程一樣,就完全被Python線程調度機制所控制了。
需要注意的是,PyThread_start_new_thread是在主線程中執(zhí)行的,而從bootstrap開始,則是在子線程中執(zhí)行的。其中涉及線程銷毀的動作,如PyThreadState_ DeleteCurrent等,將在后續(xù)的部分剖析。到了這里,讀者可能有些疑惑了,我們花費了大量篇幅剖析的線程狀態(tài)對象鏈表似乎沒有什么用啊。其實不然,試想一下,當線程調度發(fā)生時。
在Python一級,需要通過之前剖析過的PyTrheadState_Swap函數切換當前的線程狀態(tài)對象,這時候就需要根據線程id從線程狀態(tài)對象鏈表中獲取線程對象了。事實上,在Python內部的許多API中,比如PyGILState_Ensure等等中,都會涉及這個鏈表,這些API在C與Python交互時可能被大量調用,有興趣的讀者可以自行深入探索一下。
【編輯推薦】
- 漫談Python 源代碼編制技巧
- 簡單易于操作的Python 工具詳解
- 有關Python應用領域進行說明介紹
- PythonAndroid面向對象的編程——Python應用程序
- 如何使用Python模塊解析配置文件 ?
分享名稱:Python主線程與Python子線程之間區(qū)別之談
文章地址:http://m.fisionsoft.com.cn/article/dhgeiji.html


咨詢
建站咨詢
