- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
异步 docs阅读:
Most asyncio objects are not thread safe. You should only worry if you access objects outside the event loop.
有人可以对此进行解释或举例说明滥用 asyncio 如何导致对线程之间共享的对象进行不同步写入吗?我认为 GIL 意味着一次只有一个线程可以运行解释器,因此解释器中发生的事件(如读取和写入 Python 对象)在线程之间很容易同步。
上面引文中的第二句话听起来像是一个线索,但我不确定该怎么做。
我猜一个线程总是会通过释放 GIL 并决定写入 Python 对象而造成严重破坏,但这并不是 asyncio 特有的,所以我认为这不是文档在这里所指的。
这可能是因为 asyncio PEP 为某些 asyncio 对象保留了非线程安全的选项,即使目前 CPython 中的实现恰好是线程安全的?
最佳答案
其实,不,每个线程就是那个,解释器的一个新线程。
它是由 OS 管理的真正线程,而不是仅供 Python 虚拟机中的 Python 代码使用的内部管理线程。
需要 GIL 来防止非常基于操作系统的线程弄乱 Python 对象。
想象一个线程在一个 CPU 上,另一个线程在另一个 CPU 上。纯并行线程,用汇编语言编写。两者同时试图更改注册表值。根本不是理想的情况。访问相同内存位置的汇编指令最终会争先恐后地决定何时何地移动什么。这样的 Action 最终的结果很容易导致segmentation fault。好吧,如果我们用 C 编写,C 控制那部分,所以这不会发生在 C 代码中。 GIL 在 C 级别对 Python 代码执行相同的操作。因此,实现 Python 对象的代码在更改它们时不会失去其原子性。想象一下,一个线程将一个值插入到一个列表中,该列表正在另一个线程中向下移动,因为另一个线程从中删除了一些元素。如果没有 GIL,这会崩溃。
GIL 对线程内代码的原子性不做任何处理。它仅用于内部内存管理。
即使你有像 deque() 这样的线程安全对象,如果你一次对它执行多个操作,没有额外的锁,你可以从插入在两者之间的另一个线程获得结果。糟糕,问题出现了!
假设一个线程从堆栈中取出一个对象,检查它的某些内容,如果条件正确则将其删除。
stack = [2,3,4,5,6,7,8]
def thread1 ():
while 1:
v = stack[0]
sleep(0.001)
if v%2==0: del stack[0]
sleep(0.001)
当然,这是愚蠢的,应该使用 stack.pop(0) 来避免这种情况。但这是一个例子。
让另一个线程每 0.002 秒添加一次堆栈:
def thread2 ():
while 1:
stack.insert(0, stack[-1]+1)
sleep(0.002)
现在如果你这样做:
thread(thread2,())
sleep(1)
thread(thread1,())
虽然不太可能,但会有一个时刻,thread2() 会尝试恰好在 thread1() 的检索和删除之间堆叠新项目。因此,thread1() 将删除新添加的项目而不是正在检查的项目。结果不符合我们的意愿。因此,GIL 不控制我们在线程中所做的事情,只是在更基本的意义上控制线程对彼此所做的事情。
假设您编写了一个服务器来为某些事件购买门票。两个用户同时连接并尝试购买同一张票。如果您不小心,用户可能会一个一个坐在另一个上面。
线程安全对象是执行操作的对象,并且在第一个操作完成之前不允许另一个操作发生。
例如,如果您在一个线程中迭代 deque(),而在它中间另一个线程试图追加一些东西,append() 将阻塞,直到第一个线程完成对它的迭代。这是线程安全的。
关于python - 考虑到 GIL,asyncio 怎么可能不是线程安全的?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41459485/
我正在尝试解决 spoj 上的问题。显然,根据注释,输入行以 \r\n 结尾。我从之前的问题中了解到的关于 \r\n 的信息是它是 Windows 的东西。我想知道的是如何考虑它。目前我在 C++ 中
有大量对象。集合是动态的:对象可以随时添加或删除。我们称对象总数为 N。 每个对象都有两个属性:上次更新的质量 (M) 和时间 (T)。 每隔 X 分钟,应从中选择一小批进行处理,这会将它们的 T 更
我有一张带有免费赠品的注册表,但有人滥用它注册为 jo+1@gmail.com jo+2@gmail.com 有没有我可以在我的电子邮件记录上运行的标准算法来验证这种情况? 我可以在其中获取 jo@
我正在管理一个已建立的网站,该网站目前正在升级(完全重新更换),但我担心我会失去所有的 Google 索引(也就是说,会有很多页面在谷歌的索引中,它将不再存在于那个地方)。 上次我升级一个(不同的)网
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎与 help center 中定义的范围内的编程无关。 . 关闭 6 个月前。 Improv
思考横向拿着 iPhone X。 由于安全区域看起来是对称的,所以安全区域的插入在左侧和右侧一样大,无论缺口在哪一侧。 假设我们希望将安全区域进一步延伸到没有凹口的那一侧的屏幕边缘。有人对此有好的方法
Haskell 通常被视为纯函数式语言的示例。鉴于 System.IO.Unsafe.unsafePerformIO 的存在,这如何合理? 编辑:我认为“纯功能”意味着不可能将不纯的代码引入到程序的功
我创建了两个 HashMap ,我想在同一个 for 循环中迭代它们。 HashMap hashmapA = new HashMap<>(); HashMap hashmapB = new HashM
在阅读了 Venkat Subramaniam 所著的《Java 函数式编程》第 106-108 页中的“昂贵资源的延迟初始化”之后,发现很难理解此代码片段的技巧 我的理解:变量heavy上课Hold
我有一个模型 A,它包含一个通用外键关系,并限制选择其他 3 个模型(将它们视为 B、C和 D) 在同一个应用程序中。我知道我们不能使用 filter 或 get 或任何其他查询集操作的通用外键的局限
我正计划制作一个 最终 使用 Cassandra 的网络应用程序.但现在我没有它,也没有服务器来运行它,我想从使用 MySQL 开始。 问题是,是否有可能以这样一种方式构建数据和查询,以便稍后将其移植
所以我刚开始,学习不同的教程并通过征服基础知识来挑战自己。我知道这对大多数人来说可能看起来很低调,但要温和一点,我对此有点陌生。 我尝试使用媒体查询 4,例如 @media (30em <= widt
我需要遍历 .csv,直到名为“seguir”的 bool 值变为 False。问题是 takewhile 丢弃了最后一次迭代和值,我需要它,因为该行中的 .csv 值与我的问题相关。 for lin
我试试这个: Stopwatch timer = new Stopwatch(); timer.Start(); for (i = 0; i < 100; i++) { //do here s
我看过这个讨论 map 的帖子:Map function in MATLAB? 因此,我想知道是否有类似的方法来考虑索引。比如如果我想做类似 (x_i)^i 的事情。 非常感谢! 最佳答案 一个例子:
异步 docs阅读: Most asyncio objects are not thread safe. You should only worry if you access objects o
背景 假设我有一个 Google map View ,另一个 View 在它上面,覆盖了它的一部分,隐藏了 map 的一些内容。 问题 我需要制作 map 的“相机”,聚焦并在坐标上有一个标记,但让它
有什么方法可以找出目录占用的原始 HDFS 空间。据我所知 hdfs dfs -du -s/dir 显示 /dir 大小,不考虑内部文件的复制。 最佳答案 运行命令 hadoop fsck/dir 并
我正在用 Java 编写程序,我需要确定某个日期是否是周末。然而,我需要考虑到在不同的国家周末是在不同的日子,例如在以色列是周五和周六,而在一些伊斯兰国家是周四和周五。更多详情可以查看this Wik
我将HttpRequestBase包装为HttpRequestMessage,这要求将所有 header 都复制过来。但是,由于HttpRequestMessage特定于哪些 header 应用于请求
我是一名优秀的程序员,十分优秀!