Python主线程与Python子线程之间区别之谈

开发 后端
下面您所看的文章主要展示了从主线程开始创建Python子线程,以及主线程到子线程进入Python解释器的所有函数调用。

Python子线程在创建自身的线程状态对象后,会通过_PyGILState_NoteThreadState这个语句将这个对象放入到线程状态对象链表中,当前活动的Python子线程不一定是获得了GIL的线程。

在thread1.py中主线程现在是获得了GIL的,但是子线程到现在还没有申请GIL,自然也不会将自身挂起。由于主线程和子线程都是Win32的原生线程。所以操作系统可能在主线程和Python子线程之间切换。我们在这里要着重指出操作系统级的线程调度和Python级的线程调度是不同的。

Python级的线程调度一定意味着GIL拥有权的易手,而操作系统级的线程调度并不一定意味着GIL的易手,当所有的线程都完成了初始化动作之后。操作系统的线程调度和Python的线程调度才会同一。那时,Python的线程调度会迫使当前活动线程释放GIL,而这一操作会触发GIL中维护的Event内核对象。

这个触发又进而触发操作系统的线程调度。而在线程的初始化完成之前,在Python线程调度和操作系统线程调度之间并没有这样的因果关系。显示了GIL在Python级线程调度与操作系统级线程调度之间所起的桥梁作用。

前面我们已经剖析过PyEval_AcquireThread的代码,在PyEval_AcquireThread中,子线程进行了***的冲刺,它要生存,要执行,于是它开始通过PyThread_acquire_ lock争取GIL。到了这一步。

Python子线程将自己挂起,操作系统的线程调度机制再也不能靠自身的力量将其唤醒,只有等待Python的线程调度机制强迫主线程放弃GIL后。子线程才会被唤醒;而子线程被唤醒之后,主线程却又陷入了苦苦地等待中,同样苦苦地守望着Python强迫子线程放弃GIL的那一刻。

子线程被Python的线程调度机制唤醒之后,它所作的***件事就是通过PyThreadState_Swap将Python维护的当前线程状态对象设置为其自身的状态对象,一如操作系统的进程上下文环境恢复一样。

现在我们的Python子线程开始等待GIL,但是注意,线程的初始化还没有真正完成,因为子线程还没有顺利进入字节码解释器。当Python线程调度将子线程唤醒之后。子线程将回到t_bootstrap中。

并进入PyEval_CallObjectWithKeywords,从这里一直往前,最终将调用PyEval_EvalFrameEx,进入解释器。到了那个时候,Python子线程和主线程一样,就完全被Python线程调度机制所控制了。

需要注意的是,PyThread_start_new_thread是在主线程中执行的,而从bootstrap开始,则是在子线程中执行的。其中涉及线程销毁的动作,如PyThreadState_ DeleteCurrent等,将在后续的部分剖析。到了这里,读者可能有些疑惑了,我们花费了大量篇幅剖析的线程状态对象链表似乎没有什么用啊。其实不然,试想一下,当线程调度发生时。

在Python一级,需要通过之前剖析过的PyTrheadState_Swap函数切换当前的线程状态对象,这时候就需要根据线程id从线程状态对象链表中获取线程对象了。事实上,在Python内部的许多API中,比如PyGILState_Ensure等等中,都会涉及这个链表,这些API在C与Python交互时可能被大量调用,有兴趣的读者可以自行深入探索一下。

【编辑推荐】

  1. 漫谈Python 源代码编制技巧
  2. 简单易于操作的Python 工具详解
  3. 有关Python应用领域进行说明介绍
  4. PythonAndroid面向对象的编程——Python应用程序
  5. 如何使用Python模块解析配置文件 ?
责任编辑:chenqingxiang 来源: CSDN
相关推荐

2010-02-24 11:19:00

Python主线程

2010-03-16 17:16:38

Java多线程

2020-12-21 06:18:15

Android线程主线程

2010-02-01 17:33:24

Python主线程

2010-02-02 16:47:12

Python主线程

2010-02-26 15:37:11

Python主线程

2010-03-10 19:34:45

Python主线程

2017-01-10 13:39:57

Python线程池进程池

2010-03-18 16:02:09

python 多线程

2011-06-22 16:02:37

Qt 多线程 重入

2021-12-26 18:22:30

Java线程多线程

2018-07-06 14:00:55

Linux进程线程

2021-08-12 16:28:10

AndroidHandleLooper

2010-01-21 11:25:44

linux多线程线程资源

2012-05-14 17:09:05

iOS

2011-06-22 15:42:18

QT 信号

2021-04-08 10:51:10

主线程子线程Thread

2018-11-28 09:53:50

游戏服务器线程

2010-02-03 16:15:05

Python语言

2021-12-14 08:28:08

Java多线程线程
点赞
收藏

51CTO技术栈公众号