环球视讯!它来了!真正的 python 多线程
哈喽大家好,我是咸鱼
几天前,IBM 工程师 Martin Heinz 发文表示 python 3.12 版本回引入"Per-Interpreter GIL”,有了这个 Per-Interpreter 全局解释器锁,python 就能实现真正意义上的并行/并发
我们知道,python 的多线程/进程并不是真正意义上的多线程/进程,这是因为 python GIL (Global Interpreter Lock)导致的
【资料图】
而即将发布的 Python 3.12 中引入了名为 "Per-Interpreter GIL" 的新特性,能够实现真正的并发
接下来我们来看下这篇文章,原文链接如下:
https://martinheinz.dev/blog/97
译文
Python 到现在已经 32 岁了,但它到现在还没有实现适当的、真正的并发/并行
由于将在 Python 3.12 (预计 2023 年 10 月发布)中引入 “Per-Interpreter GIL”(全局解释器锁),这种情况将会被改变
虽然距离 python 3.12 的发布还有几个月的时间,但是相关代码已经实现了。所以让我们提前来了解一下如何使用子解释器 API(ub-interpreters API) 来编写出真正的并发Python代码
子解释器(Sub-Interpreters)
我们首先来看下这个 “Per-Interpreter GIL” 是如何解决 Python 缺失适当并发性这个问题的
简单来讲,GIL(全局解释器锁)是一个互斥锁,它只允许一个线程控制 Python 解释器(某个线程想要执行,必须要先拿到 GIL ,在一个 python 解释器里面,GIL 只有一个,拿不到 GIL 的就不允许执行)
这就意味着即使你在 Python 中创建多个线程,也只会有一个线程在运行
随着 “Per-Interpreter GIL” 的引用,单个 python 解释器不再共享同一个 GIL。这种隔离级别允许每个子 python 解释器真正地并发运行
这意味着我们可以通过生成额外的子解释器来绕过 Python 的并发限制,其中每个子解释器都有自己的GIL(拿到一个 GIL 锁)
更详细的说明请参见 PEP 684,该文档描述了此功能/更改:https://peps.python.org/pep-0684/#per-interpreter-state
如何安装
想要使用这个新功能,我们需要安装最新的 python 版本,这需要源码编译安装
# https://devguide.python.org/getting-started/setup-building/#unix-compilinggit clone https://github.com/python/cpython.gitcd cpython./configure --enable-optimizations --prefix=$(pwd)/python-3.12make -s -j2./python# Python 3.12.0a7+ (heads/main:22f3425c3d, May 10 2023, 12:52:07) [GCC 11.3.0] on linux# Type "help", "copyright", "credits" or "license" for more information.C-API 在哪里
现在我们已经安装好了最新版本,那么我们该如何使用子解释器呢?我们可以直接通过 import来导入吗?不幸的是,还不能
正如 PEP-684 中指出的: ...this is an advanced feature meant for a narrow set of users of the C-API.
Per-Interpreter GIL 的特性目前只能通过 C-API 使用,还没有直接的接口供开发人员使用
接口预计会在 PEP 554中出现,如果大家能够接受,它应该会在 Python 3.13 中出现,在这个版本出现之前,我们必须自己想办法来实现子解释器
虽然还没有相关文档,也没有相关模块可以导入,但 CPython 代码库中有一些代码段向我们展示了如何使用它:
- 方法一:我们可以使用
_xxsubinterpreters模块(因为是通过 C 实现的,所以命名比较奇怪,而且在 python 中不能够简单地去检查代码) - 方法二:可以使用 CPython 的 test 模块,该模块具有用于测试的示例 Interpreter(和 Channel)类
# Choose one of these:import _xxsubinterpreters as interpretersfrom test.support import interpreters通常情况下我们一般用上面的第二种方法来实现
我们已经找到了子解释器,但我们还需要通过 test 模块去借用一些辅助函数,以便将代码传递给子解释器,辅助函数如下
from textwrap import dedentimport os# https://github.com/python/cpython/blob/# 15665d896bae9c3d8b60bd7210ac1b7dc533b093/Lib/test/test__xxsubinterpreters.py#L75def _captured_script(script): r, w = os.pipe() indented = script.replace("\n", "\n ") wrapped = dedent(f""" import contextlib with open({w}, "w", encoding="utf-8") as spipe: with contextlib.redirect_stdout(spipe): {indented} """) return wrapped, open(r, encoding="utf-8")def _run_output(interp, request, channels=None): script, rpipe = _captured_script(request) with rpipe: interp.run(script, channels=channels) return rpipe.read()将 interpreters模块与上面的辅助函数组合在一起,便可以生成第一个子解释器:
from test.support import interpretersmain = interpreters.get_main()print(f"Main interpreter ID: {main}")# Main interpreter ID: Interpreter(id=0, isolated=None)interp = interpreters.create()print(f"Sub-interpreter: {interp}")# Sub-interpreter: Interpreter(id=1, isolated=True)# https://github.com/python/cpython/blob/# 15665d896bae9c3d8b60bd7210ac1b7dc533b093/Lib/test/test__xxsubinterpreters.py#L236code = dedent(""" from test.support import interpreters cur = interpreters.get_current() print(cur.id) """)out = _run_output(interp, code)print(f"All Interpreters: {interpreters.list_all()}")# All Interpreters: [Interpreter(id=0, isolated=None), Interpreter(id=1, isolated=None)]print(f"Output: {out}") # Result of "print(cur.id)"# Output: 1生成和运行新解释器的一种方法是使用 create()函数,然后将解释器与我们想要执行的代码一起传递给 _run_output()辅助函数
还有一种更简单的方法,如下所示
interp = interpreters.create()interp.run(code)直接使用 interpreters模块的 run方法。
但如果我们运行上面这两段代码时,会收到以下报错
Fatal Python error: PyInterpreterState_Delete: remaining subinterpretersPython runtime state: finalizing (tstate=0x000055b5926bf398)为了避免这个报错,我们还需要清理一些悬挂的解释器:
def cleanup_interpreters(): for i in interpreters.list_all(): if i.id == 0: # main continue try: print(f"Cleaning up interpreter: {i}") i.close() except RuntimeError: pass # already destroyedcleanup_interpreters()# Cleaning up interpreter: Interpreter(id=1, isolated=None)# Cleaning up interpreter: Interpreter(id=2, isolated=None)线程
虽然使用上面的辅助函数运行代码是可行的,但在 threading模块中使用熟悉的接口可能会更方便
import threadingdef run_in_thread(): t = threading.Thread(target=interpreters.create) print(t) t.start() print(t) t.join() print(t)run_in_thread()run_in_thread()# # # # # # 我们通过把 interpreters.create函数传递给Thread,它会自动在线程内部生成新的子解释器
我们也可以结合这两种方法,并将辅助函数传递给 threading.Thread:
import timedef run_in_thread(): interp = interpreters.create(isolated=True) t = threading.Thread(target=_run_output, args=(interp, dedent(""" import _xxsubinterpreters as _interpreters cur = _interpreters.get_current() import time time.sleep(2) # Can"t print from here, won"t bubble-up to main interpreter assert isinstance(cur, _interpreters.InterpreterID) """))) print(f"Created Thread: {t}") t.start() return tt1 = run_in_thread()print(f"First running Thread: {t1}")t2 = run_in_thread()print(f"Second running Thread: {t2}")time.sleep(4) # Need to sleep to give Threads time to completecleanup_interpreters()上面的代码中演示了如何使用 _xxsubinterpreters模块来实现 (方法一)
我们还在每个线程中休眠 2 秒来模拟“工作”状态
请注意,我们甚至不必调用 join() 函数等待线程完成,只需在线程完成时清理解释器即可
Channels
如果我们进一步挖掘 CPython test模块,我们还会发现 RecvChannel 和 SendChannel 类的实现类似于 Golang 中已知的通道
# https://github.com/python/cpython/blob/# 15665d896bae9c3d8b60bd7210ac1b7dc533b093/Lib/test/test_interpreters.py#L583r, s = interpreters.create_channel()print(f"Channel: {r}, {s}")# Channel: RecvChannel(id=0), SendChannel(id=0)orig = b"spam"s.send_nowait(orig)obj = r.recv()print(f"Received: {obj}")# Received: b"spam"cleanup_interpreters()# Need clean up, otherwise:# free(): invalid pointer# Aborted (core dumped)上面的例子介绍了如何创建一个接收端通道(r)和发送端通道(s),然后我们使用 send_nowait方法将数据发送,通过 recv方法来接收数据
这个通道实际上只是另一个解释器,和以前一样,我们需要在处理完它之后进行清理
Digging Deeper
如果我们想要修改或者调整子解释器的选项(这些选项通常在 C 代码中设置),我们可以使用
test.support模块中的代码,具体来说是run_in_subinterp_with_config
import test.supportdef run_in_thread(script): test.support.run_in_subinterp_with_config( script, use_main_obmalloc=True, allow_fork=True, allow_exec=True, allow_threads=True, allow_daemon_threads=False, check_multi_interp_extensions=False, own_gil=True, )code = dedent(f""" from test.support import interpreters cur = interpreters.get_current() print(cur) """)run_in_thread(code)# Interpreter(id=7, isolated=None)run_in_thread(code)# Interpreter(id=8, isolated=None)上面这个run_in_subinterp_with_config函数是 C 函数的 Python API。它提供了一些子解释器选项,如 own_gil,指定子解释器是否应该拥有自己的 GIL
标签:
环球视讯!它来了!真正的 python 多线程
2023-05-25
对流烹饪和烧烤有什么区别?_妈妈过生日送什么礼物手工制作的 天天简讯
2023-05-25
从居住品质评判房子标准|每日精选
2023-05-25
世界球精选!cad闭合快捷键命令是什么(cad闭合快捷键)
2023-05-25
对同事的工作评价缺点_对同事的工作评价
2023-05-25
【天天时快讯】比利时记者称不希望安特卫普夺冠,遭球迷怒斥炮轰
2023-05-25
拉维克的诱饵在哪_WOW 拉克维的气味请问这个说打雌性恐龙的任务怎么做 前沿热点
2023-05-25
吉林省肉牛活体贷案例入选地方金融支农十大典型案例
2023-05-25
闹闹女巫 0525十二星座今日运势 消息
2023-05-25
湖北日报评论员:城市治理现代化任重道远|环球速递
2023-05-25
对流烹饪和烧烤有什么区别?_妈妈过生日送什么礼物手工制作的 天天简讯
从居住品质评判房子标准|每日精选
世界球精选!cad闭合快捷键命令是什么(cad闭合快捷键)
对同事的工作评价缺点_对同事的工作评价
【天天时快讯】比利时记者称不希望安特卫普夺冠,遭球迷怒斥炮轰
拉维克的诱饵在哪_WOW 拉克维的气味请问这个说打雌性恐龙的任务怎么做 前沿热点
吉林省肉牛活体贷案例入选地方金融支农十大典型案例
闹闹女巫 0525十二星座今日运势 消息
湖北日报评论员:城市治理现代化任重道远|环球速递
停息挂账还清后信用卡还能使用吗?还清后信用会消除吗?-天天新动态
国家发改委辟谣!金融圈突发!民生银行、海通证券紧急声明-世界播报
陕西勉县夏播忙-全球观速讯
5月24日基金净值:交银创新领航混合最新净值1.2445,跌0.69%
苏丹冲突双方分别发表声明 指责对方违反停火协议 全球看点
世界视点!AI前哨 | 文心一言市场负责人张全文:也是醉了 别动不动碰瓷
慕容无言(关于慕容无言介绍) 世界快资讯
华硕键盘背光灯驱动叫什么(华硕键盘背光灯驱动)|天天快资讯
纳思达2023校园招聘文案策划(附报名入口) 世界快资讯
搭建三级调查体系,武汉市洪山区提升“双评议”质效
全球热讯:预算10万左右适合女性开的车_预算10万左右什么车性价比比较高
2019会计初级考试时间下半年_2019会计初级考试时间
今日聚焦!易次元仙途新手攻略_仙途新手卡领取
全球看热讯:陕西中医药大学第二附属医院:“移花接木”脚趾变成大拇指
明日起天气转折!四川东北部等部分地区将迎强降雨
【世界新视野】TCL中环:与Vision Industries Company签署合作条款清单
世界微速讯:为什么建设银行限额1000 可能是这些原因
血亏!重庆棕榈泉豪宅二手跌破9000一平!土豪竞相夺路而逃? 当前快报
众多“神器”亮相中关村论坛,记者带你抢先看!|天天热资讯
科雷傲内饰改装推荐
- 收到传票小店被起诉商标侵权不去都会有什么后果|新动态
- 印度民众抓紧时间用2000卢比纸币购物 以免9月后无法流通-全球热点评
- 《原神》艾尔海森前期四星圣遗物搭配推荐
- 环球最新:上海即日起开展住宅小区建筑外墙高空坠物隐患专项排查整治“百日行动”
- 罗罗公司成功完成UltraFan技术演示样机首次试车
- 精彩的成语故事大全-最精彩的成语故事
- 原神宵宫最强配队推荐 3.7版宵宫配队大全[多图]
- 北野武自导自演战国题材电影《首》亮相戛纳首映单元 环球快播报
- 安科生物:5月23日融资买入534.38万元,融资融券余额5.08亿元 焦点快看
- 四大坚果是什么树_四大坚果是什么
- 河化股份5月23日快速回调_世界聚看点
- 这个品种表现强势!市场人士:需注意风险控制
- 【独家焦点】白色蕾丝三角裤紧紧包裹住隆起的 三角裤后的黑毛毛
- 当前简讯:中专毕业证怎么查验真假_中专毕业证怎么查真假
- 环球时讯:虹猫蓝兔七侠传吧宏梦_虹猫蓝兔七侠传吧
- 赛尔号最新特性大全_赛尔号特性大全
- 小儿肠梗阻的症状及治疗_小儿肠梗阻的表现
- 奇数页眉和偶数页眉分别怎么设置页码奇数页和偶数页不同_奇数页眉和偶数页眉分别怎么设置
- 快看:花字成语大全四个字_花字成语
- 天天快资讯丨“菜单式”志愿服务,苏州太平街道黎明村实现为民服务“零距离”
- 记者:维尼修斯今天没有参加训练,只在健身房里工作 全球新动态
- 中超又成拳击台!申花门将复制贾德松,拳击对手染红,或被禁赛5场|世界速读
- 天天看点:荣誉之路,新星闪耀!2023HEAD超新星冠军赛广州站收官
- DNF: 1亿氪金拿亚军,旭旭宝宝不孤单|环球速递
- 青春入“牟”Happy一夏!这个暑假,持一张学生证走遍中牟
- 新鲜蛋如何保存 新鲜蛋的保存方法-天天热推荐
- 塞尔达传说王国之泪讲究的男人多当茨挑战视频攻略
- 世界观焦点:鸿泉物联接待信达证券等多家机构调研
- 魏牌新摩卡开启预定!配联动大四屏 每公里油耗4毛钱
- 世界滚动:get什么意思网络上的_网络流行语get什么意思
- 什么是消费者的公平交易权?生活中消费者遇到价格不公平可以要求退款吗?
- 世界信息:宝可梦大探险木守宫配方(宝可梦大师木守宫石盘)
- 水果新手怎么进货_水果批发新手如何进货简介介绍|世界资讯
- 搜狐汽车全球快讯 | 比亚迪或考虑在法国建厂 比亚迪:正评估建厂可行性
- 焦点播报:苹果Apple Watch产量占比40% 立讯成2022年下半年最大的智能手表外包制造商
- 耐力板规格尺寸价格_耐力板规格
- 当前简讯:中小盘定增并购月报:竞价定增折价回升 重点跟踪怡合达、奥来德
- 【凭栏吟唱】黎水行 环球热头条
- 快手-W绩后高开逾3% 一季度营收同比增近20% 品牌广告收入连续两季度同比增长超20%|环球焦点
- 天天关注:肝肾亏虚能导致脚凉吗
- 全国3053个执法办案中心投入使用
- 世界焦点!手机怎么设置卫星地图_手机怎么设置卫星定位
- 三晋大地高质量发展动能强劲
- *ST辅仁实控人独家回应:我没失联过,现在最重要的是配合完成企业重整|当前关注
- 全球热门:国际生物多样性日丨一群国宝·一位青年·石山精灵
- 家用咖啡机十大品牌排行榜_家用咖啡机十大品牌排行榜_热点在线
- 焦点速递!鞋子泛硝怎么处理_墙面起硝怎么彻底处理
- 宁波旭力金属制品有限公司
- 【当前独家】供地热丨高新33宗住宅用地上新,吸引60余家房企关注,未来房价…
- 亮相马来西亚,空军八一飞行表演队进行首场适应性训练|快讯
