列表是线程安全的吗?

2024-12-27 08:47:00
admin
原创
134
摘要:问题描述:我注意到,通常建议使用具有多个线程的队列,而不是列表和.pop()。这是因为列表不是线程安全的,还是出于其他原因?解决方案 1:列表本身是线程安全的。在 CPython 中,GIL 可防止对列表的并发访问,而其他实现会小心地为其列表实现使用细粒度锁或同步数据类型。但是,虽然列表本身不会因尝试并发访问...

问题描述:

我注意到,通常建议使用具有多个线程的队列,而不是列表和.pop()。这是因为列表不是线程安全的,还是出于其他原因?


解决方案 1:

列表本身是线程安全的。在 CPython 中,GIL 可防止对列表的并发访问,而其他实现会小心地为其列表实现使用细粒度锁或同步数据类型。但是,虽然列表本身不会因尝试并发访问而损坏,但列表的数据不受保护。例如:

L[0] += 1

如果另一个线程执行了相同的操作,则不能保证 L[0] 实际上会增加 1,因为+=这不是原子操作。(Python 中只有极少数操作实际上是原子的,因为大多数操作都可能导致调用任意 Python 代码。)您应该使用队列,因为如果您只是使用不受保护的列表,您可能会因为竞争条件而获取或删除错误的项目

解决方案 2:

为了澄清托马斯的出色回答中的一点,应该提到线程append() 安全

这是因为,我们不必担心一旦写入数据,读取的数据就会位于同一位置。该操作不会读取数据,只会将数据写入列表。append()

解决方案 3:

以下是操作示例的全面但非详尽列表list,以及它们是否是线程安全的。希望在这里obj in a_list得到有关语言构造的答案。

解决方案 4:

我最近遇到过这种情况,我需要在一个线程中连续向列表添加内容,循环遍历项目并检查项目是否已准备好,在我的情况下它是一个 AsyncResult,并且只有当它准备好时才将其从列表中删除。我找不到任何能清楚地说明我的问题的示例这里有一个示例,演示了在一个线程中连续向列表添加内容并在另一个线程中连续从同一列表中删除内容。有缺陷的版本在较小的数字上很容易运行,但保持数字足够大并运行几次,你就会看到错误

有缺陷的版本

import threading
import time

# Change this number as you please, bigger numbers will get the error quickly
count = 1000
l = []

def add():
    for i in range(count):
        l.append(i)
        time.sleep(0.0001)

def remove():
    for i in range(count):
        l.remove(i)
        time.sleep(0.0001)


t1 = threading.Thread(target=add)
t2 = threading.Thread(target=remove)
t1.start()
t2.start()
t1.join()
t2.join()

print(l)

ERROR 时输出

Exception in thread Thread-63:
Traceback (most recent call last):
  File "/Users/zup/.pyenv/versions/3.6.8/lib/python3.6/threading.py", line 916, in _bootstrap_inner
    self.run()
  File "/Users/zup/.pyenv/versions/3.6.8/lib/python3.6/threading.py", line 864, in run
    self._target(*self._args, **self._kwargs)
  File "<ipython-input-30-ecfbac1c776f>", line 13, in remove
    l.remove(i)
ValueError: list.remove(x): x not in list

使用锁的版本

import threading
import time
count = 1000
l = []
lock = threading.RLock()
def add():
    with lock:
        for i in range(count):
            l.append(i)
            time.sleep(0.0001)

def remove():
    with lock:
        for i in range(count):
            l.remove(i)
            time.sleep(0.0001)


t1 = threading.Thread(target=add)
t2 = threading.Thread(target=remove)
t1.start()
t2.start()
t1.join()
t2.join()

print(l)

输出

[] # Empty list

结论

正如前面的答案中提到的那样,虽然从列表本身附加或弹出元素的操作是线程安全的,但当你在一个线程中附加元素并在另一个线程中弹出元素时,就不是线程安全的了

相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   1565  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1354  
  信创国产芯片作为信息技术创新的核心领域,对于推动国家自主可控生态建设具有至关重要的意义。在全球科技竞争日益激烈的背景下,实现信息技术的自主可控,摆脱对国外技术的依赖,已成为保障国家信息安全和产业可持续发展的关键。国产芯片作为信创产业的基石,其发展水平直接影响着整个信创生态的构建与完善。通过不断提升国产芯片的技术实力、产...
国产信创系统   21  
  信创生态建设旨在实现信息技术领域的自主创新和安全可控,涵盖了从硬件到软件的全产业链。随着数字化转型的加速,信创生态建设的重要性日益凸显,它不仅关乎国家的信息安全,更是推动产业升级和经济高质量发展的关键力量。然而,在推进信创生态建设的过程中,面临着诸多复杂且严峻的挑战,需要深入剖析并寻找切实可行的解决方案。技术创新难题技...
信创操作系统   27  
  信创产业作为国家信息技术创新发展的重要领域,对于保障国家信息安全、推动产业升级具有关键意义。而国产芯片作为信创产业的核心基石,其研发进展备受关注。在信创国产芯片的研发征程中,面临着诸多复杂且艰巨的难点,这些难点犹如一道道关卡,阻碍着国产芯片的快速发展。然而,科研人员和相关企业并未退缩,积极探索并提出了一系列切实可行的解...
国产化替代产品目录   28  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用