国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

如何正確區分Python線程

開發 后端
怎么區分Python線程是屬于哪個狀態對象呢,幸好我們還有線程id呢,ID存儲的正是各個線程的id,根據這個有效ID,可以區分不同的Python線程。

在Python語言中Python線程可以從這里開始與主線程對GIL的競爭,在t_bootstrap中,申請完了GIL,也就是說子線程也就獲得了GIL,使其始終保存著活動線程的狀態對象。

當PyEval_AcquireThread結束之后,子線程也就獲得了GIL,并且做好了一切執行的準備。接下來子線程通過PyEval_ CallObjectWithKeywords,將最終調用我們已經非常熟悉的PyEval_EvalFrameEx。

也就是Python的字節碼執行引擎。傳遞進PyEval_CallObjectWithKeywords的boot->func是一PyFunctionObject對象,正是therad1.py中定義的threadProc編譯后的結果。在PyEval_CallObjectWithKeywords結束之后,子線程將釋放GIL,并完成銷毀線程的所有掃尾工作,到了這里,子線程就結束了。

從t_bootstrap的代碼看上去,似乎子線程會一直執行,直到子線程的所有計算都完成,才會通過PyThreadState_DeleteCurrent釋放GIL。如此一來,那主線程豈非一直都會處于等待GIL的狀態?如果真是這樣,那Python線程顯然就不可能支持多線程機制了。

實際上在PyEval_EvalFrameEx中,圖15-2中顯示的Python內部維護的那個模擬時鐘中斷會不斷地激活線程的調度機制,在子線程和主線程之間不斷地進行切換。從而真正實現多線程機制,當然,這一點我們將在后面詳細剖析?,F在我們感興趣的是子線程在PyEval_AcquireThreade中到底做了什么。

到這里,了解了PyEval_AcquireThread,似乎創建線程的機制都清晰了。但實際上,有一個非常重要的機制——線程狀態保護機制——隱藏在了一個毫不起眼的地方:PyThreadState_New。

  1. [threadmodule.c]  
  2.  
  3. static PyObject* thread_PyThread_start_new_thread(PyObject *self, PyObject  
  4.  
  5.   *fargs)  
  6.  
  7. {  
  8.  
  9.     PyObject *func, *args, *keyw = NULL;  
  10.  
  11.     struct bootstate *boot;  
  12.  
  13.     long ident;  
  14.  
  15.     PyArg_UnpackTuple(fargs, "start_new_thread", 2, 3, &func, &args, &keyw);  
  16.  
  17.     //[1]:創建bootstate結構  
  18.  
  19.     boot = PyMem_NEW(struct bootstate, 1);  
  20.  
  21.     boot->interp = PyThreadState_GET()->interp;  
  22.  
  23.     boot->funcfunc = func;  
  24.  
  25.     boot->argsargs = args;  
  26.  
  27.     boot->keywkeyw = keyw;  
  28.  
  29.     //[2]:初始化多線程環境  
  30.  
  31.     PyEval_InitThreads(); /* Start the interpreter's thread-awareness */  
  32.  
  33.     //[3]:創建線程  
  34.  
  35.     ident = PyThread_start_new_thread(t_bootstrap, (void*) boot);  
  36.  
  37.     return PyInt_FromLong(ident);  
  38.  
  39. [thread.c]  
  40.  
  41. /* Support for runtime thread stack size tuning.  
  42.  
  43.    A value of 0 means using the platform's default stack size  
  44.  
  45.    or the size specified by the THREAD_STACK_SIZE macro. */  
  46.  
  47. static size_t _pythread_stacksize = 0;  
  48.  
  49. [thread_nt.h]  
  50.  
  51. long PyThread_start_new_thread(void (*func)(void *), void *arg)  
  52.  
  53. {  
  54.  
  55.     unsigned long rv;  
  56.  
  57.     callobj obj;  
  58.  
  59.     obj.id = -1;    /* guilty until proved innocent */  
  60.  
  61.     obj.func = func;  
  62.  
  63.     obj.arg = arg;  
  64.  
  65.     obj.done = CreateSemaphore(NULL, 0, 1, NULL);  
  66.  
  67.     rv = _beginthread(bootstrap, _pythread_stacksize, &obj); /* use default stack size */  
  68.  
  69.     if (rv == (unsigned long)-1) {  
  70.  
  71.         //創建raw thread失敗  
  72.  
  73.         obj.id = -1;  
  74.  
  75.     }  
  76.  
  77.     else {  
  78.  
  79.         WaitForSingleObject(obj.done, INFINITE);  
  80.  
  81.     }  
  82.  
  83.     CloseHandle((HANDLE)obj.done);  
  84.  
  85.     return obj.id;  
  86.  
  87. }  

這個機制對于理解Python線程的創建和維護是非常關鍵的。要剖析線程狀態的保護機制,我們首先需要回顧一下線程狀態。在Python中,每一個Python線程都會有一個線程狀態對象與之關聯。

在線程狀態對象中,記錄了每一個線程所獨有的一些信息。實際上,在剖析Python的初始化過程時,我們曾經見過這個對象。每一個線程對應的線程狀態對象都保存著這個線程當前的PyFrameObject對象,線程的id這樣一些信息。有時候,線程是需要訪問這些信息的。

比如考慮一個最簡單的情形,在某種情況下,每個線程都需要訪問線程狀態對象中所保存的thread_id信息,顯然,線程A獲得的應該是A的thread_id,線程B亦然。倘若線程A獲得的是B的thread_id,那就壞菜了。這就意味著Python線程內部必須有一套機制,這套機制與操作系統管理進程的機制非常類似。

我們知道,在操作系統從進程A切換到進程B時,首先會保存進程A的上下文環境,再進行切換;當從進程B切換回進程A時,又會恢復進程A的上下文環境,這樣就保證了進程A始終是在屬于自己的上下文環境中運行。

這里的線程狀態對象就等同于進程的上下文,Python同樣會有一套存儲、恢復線程狀態對象的機制。同時,在Python內部,維護著一個全局變量:PyThreadState * _PyThread- State_Current。

當前活動線程所對應的線程狀態對象就保存在這個變量里,當Python調度線程時,會將被激活的線程所對應的線程狀態對象賦給_PyThreadState_Current,使其始終保存著活動線程的狀態對象。

這就引出了這樣的一個問題:Python如何在調度進程時,獲得被激活線程對應的狀態對象?Python內部會通過一個單向鏈表來管理所有的Python線程的狀態對象,當需要尋找一個線程對應的狀態對象時。#t#

就遍歷這個鏈表,搜索其對應的狀態對象。在此后的描述中,我們將這個鏈表稱為“狀態對象鏈表”。下面我們來看一看實現這個機制的關鍵數據結構在Python中,對于這個狀態對象鏈表的訪問,不必在GIL的保護下進行。

因為對于這個狀態對象鏈表,Python線程會創建一個獨立的鎖,專職對狀態對象鏈表進行保護。這個鎖的創建是在Python進行初始化的時候完成的。PyThread_create_key將創建一個新的key。注意,這里的key都是一個整數,而且,當PyThread_create_key***次被調用時(在_PyGILState_Init中的調用正是***次調用)。

會通過PyThread_allcate_lock創建一個keymutex。根據我們前面的分析,這個keymutex實際上和GIL一樣,都是一個PNRMUTEX結構體,而在這個結構體中,維護著一個Win32下的Event內核對象。這個keymutex的功能就是用來互斥對狀態對象鏈表的訪問。

在_PyGILState_Init中,創建的新key被Python維護的全局變量autoTLSkey接收,其中的TLS是Thread Local Store的縮寫,這個autoTLSkey將用作Python保存所有線程的狀態對象的一個參數。的key值。也就是說,狀態對象列表中所有key結構體中的key值都會是autoTLSkey。哎,那位看官說了,你看PyThread_create_key返回的是nkeys的遞增后的值啊。

就是說每create一次,得到的結果都是不同的,怎么能說所有的key都是一樣的呢?事實上,在整個Python的源碼中,PyThread_create_key只在_PyGILState_Init中被調用了,而這個_PyGILState_Init只會在Python運行時環境初始化時調用一次。

責任編輯:chenqingxiang 來源: 比特網
相關推薦

2015-05-19 16:21:05

2010-08-26 09:40:00

2010-08-25 13:13:04

2010-06-07 09:03:33

MySQL大小寫

2019-03-13 22:40:15

機器學習假設算法

2023-09-08 12:19:01

線程方法interrupt

2022-02-28 07:01:22

線程中斷interrupt

2025-02-06 03:14:38

2024-10-21 18:12:14

2025-07-29 09:06:04

2010-02-03 17:42:30

2010-02-22 14:13:38

安裝Python

2010-02-03 14:15:18

Python 開發

2010-02-22 10:06:17

Python調用

2010-02-24 15:27:26

Python數組

2021-03-15 12:23:24

Pythonyield代碼

2010-02-03 15:40:37

Python函數

2010-02-02 18:20:43

Python編寫

2018-03-30 09:21:30

程序員網絡招聘

2010-02-02 14:11:14

Python 進行編程
點贊
收藏

51CTO技術棧公眾號

99福利在线观看| 精品女人视频| 亚洲日本va中文字幕| 91九色在线观看| 色综合久久影院| 国产精品麻豆| 日韩欧美国产激情| 韩日午夜在线资源一区二区| 国产亚洲精品自在久久| 欧美成人基地| 亚洲人成绝费网站色www| 亚洲女同精品视频| 91超碰在线电影| 韩国精品免费视频| 日韩美女视频免费在线观看| 国产亚洲久久| 日韩精品免费在线| 天堂在线免费av| 91精品一区国产高清在线gif| 国产乱码精品一区二区三区日韩精品 | 国产日韩一级二级三级| 中文字幕一区视频| av7777777| 日韩片欧美片| 91丨porny丨国产入口| 精灵使的剑舞无删减版在线观看| 精品久久久久久国产91| 高清中文字幕一区二区三区| 日韩免费观看高清| av免费不卡国产观看| 欧美激情三级免费| 99精品视频播放| 人人狠狠综合久久亚洲婷婷 | 成人av免费在线看| 97精品国产99久久久久久免费| 天天av天天翘天天综合网色鬼国产| 日韩久久不卡| 一区二区三区国产好| 欧美精品自拍偷拍| 欧美色男人天堂| 一区二区国产盗摄色噜噜| 欧美精品久久久| 欧美日韩一二三四| 久久精品国产美女| 清纯唯美亚洲经典中文字幕| 日韩美女免费视频| 亚洲天堂黄色| 欧美激情综合五月色丁香小说| 亚洲精品在线免费| 欧美日韩国产免费观看| 中文字幕欧美日韩va免费视频| 尤物网在线观看| 日韩一区二区三免费高清| 在线精品一区二区| av2020不卡| 91精品国产一区二区人妖| 免费在线观看的电影网站| 在线成人中文字幕| 久久国产精品美女| 自拍偷拍亚洲综合| 日本搞黄视频| 亚洲人午夜精品| 欧美日韩激情电影| 亚洲午夜性刺激影院| 6080yy精品一区二区三区| 在线观看国产一级片| 黄色免费成人| 国内精品国语自产拍在线观看| 亚洲一区二区三区国产| 国产最新视频在线| 在线色欧美三级视频| 日韩高清欧美| 欧美一区二区三区在线免费观看| 欧美色图在线视频| 色综合五月天| 欧美日韩在线一区二区| 95在线视频| 69174成人网| 不卡一区在线观看| 黄色网免费看| 另类一区二区| 欧美猛交免费看| 欧美日韩水蜜桃| 久久精品欧美日韩精品| 久草免费在线播放| 亚洲精品国产高清久久伦理二区| 国产福利影院在线观看| 国产激情一区| 91在线视频官网| 国产视频精品xxxx| 亚洲 国产 欧美一区| 狠狠狠色丁香婷婷综合激情| 不卡的av中文字幕| 欧美日韩国产天堂| 国产免费久久| 爱情岛论坛vip永久入口| 欧美精品日韩综合在线| 亚洲国产欧美在线人成| 免费看成人午夜电影| 91视频国产观看| 伊人在我在线看导航| 国产精品91久久久| 成人av综合网| 亚洲AV无码成人精品一区| 色88888久久久久久影院野外| 成人av地址| 日本中文字幕片| 国产亚洲视频在线观看| 男人的天堂成人在线| 美脚丝袜一区二区三区在线观看| 亚洲欧洲美洲av| 亚洲精品xxx| 欧美日韩一区二区三区四区在线观看| 波多野结衣xxxx| 日韩色av导航| 国产999精品久久久久久绿帽| 福利成人导航| 亚洲图片都市激情| 国产精品99久久久| 久久电影网站| 精品久久久久久一区| 日韩欧美高清视频| 白天操夜夜操| 日日骚av一区| 成人午夜激情视频| 99热在线观看免费| 亚洲综合偷拍欧美一区色| 久久亚洲国产精品尤物| 国产高清免费在线| 欧美国产二区| 欧美精品福利在线| 精品福利樱桃av导航| 深夜福利成人| 亚洲欧美精品在线| 国产日韩欧美精品一区| 日韩久久综合| 午夜在线激情影院| 污版视频在线观看| 手机成人av在线| 日韩成人在线观看| 国内精品免费在线观看| 98在线视频| 综合分类小说区另类春色亚洲小说欧美| 日韩欧美中文在线观看| 黄大色黄女片18第一次| 国产91色在线| 免费在线激情视频| 国产偷国产偷亚洲高清97cao| 欧美一区二区在线免费观看| 久久99精品国产麻豆婷婷| www.男人的天堂| 国产精品网站视频| 91福利视频网站| 日韩亚洲国产精品| 国产精品xxxx| 亚洲精品一线二线三线无人区| 激情综合亚洲精品| 97久久精品一区二区三区的观看方式| 久久精品国产一区| 久久精品视频网| 精品国产精品国产偷麻豆| 四虎影院在线域名免费观看| 久久综合九色欧美狠狠| 国产视频精品自拍| 国产精品国产三级国产三级人妇 | 国产精品538一区二区在线| 午夜日韩成人影院| 精品日韩久久久| 亚洲自拍欧美色图| 精品成人久久av| 久久电影一区| 亚洲啊v在线免费视频| 视频一区二区三区在线看免费看| 精品国产一二| 中文字幕亚洲字幕| 国产ts一区| 日本午夜在线亚洲.国产| 亚洲中国最大av网站| 国产亚洲精品v| 精品一区视频| 91在线播放网站| 欧美乱做爰xxxⅹ久久久| 2019中文在线观看| 欧美一级高清片| 久久久精品国产免费观看同学| 91精品国产乱码久久久久久久| 中文字幕一区久| 美女av一区| 日韩一级二级| 色综合久综合久久综合久鬼88| 成人妖精视频yjsp地址| 精品国产一区二区三区久久久蜜臀| aaa在线观看| 中文字幕在线观看第三页| 国产一区二区在线网站| 久久夜色撩人精品| 宅男噜噜噜66国产日韩在线观看| 久色视频在线播放| 超碰网在线观看| 激情五月婷婷六月|