Python:在线程之间共享列表

3
我正在尝试理解线程在同一进程中运行时,它们如何相互作用和共享变量空间的微妙差别。以下代码展示了两个函数prod和consum,它们在不同的线程中运行。这两个函数都有相同的列表和锁参数:它们使用da_list来共享数据,并使用lockguy来同步/线程安全共享。
我的主要问题是:当我运行这段代码时,它打印出(1, 0, 0, 0, 0 ...)而不是我期望的(1,2,3,4,5,6...)。当我删除consum函数中的l = [0]行时,我得到了预期的行为。为什么l = [0]会弄乱它?当consum完成时,da_list应该是[0]。随后调用prod应该将da_list重置为[da_int]。谢谢你的帮助。
import threading
import time

    def prod(l,lock):
        da_int = 0
        while True:
            with lock:
                time.sleep(0.1)
                da_int += 1
                l[0] = da_int

    def consum(l,lock):

        data = ''

        while True:
            with lock:
                time.sleep(0.1)
                print(l[0])
                l = [0]

    def main():

        da_list = [0]
        lockguy = threading.Lock()


        thread1 = threading.Thread(target = prod, args=(da_list,lockguy))
        thread2 = threading.Thread(target = consum, args=(da_list,lockguy))
        thread1.daemon = True
        thread2.daemon = True
        thread1.start()
        thread2.start()

        try:
            while True:
                time.sleep(1)
        except KeyboardInterrupt:
            pass
        finally:
            print('done')




    if __name__ == '__main__':
        main()

因为您没有修改位于consum中的变量l下方的列表,而是修改了其中的本地变量l。 也就是说,l = [0] 不会 修改共享的 l 列表。 它在consum内部创建了一个新的列表,该列表与prod不共享。 请注意,例如l[:] = [] 将会 修改基础列表。 您还可以阅读以下内容:https://dev59.com/o3M_5IYBdhLWcg3wZSE6 - freakish
1个回答

5
l = [0]

您似乎对Python中的赋值方式感到困惑。您似乎认为上面这行代码修改了先前绑定到 l 的对象,但实际上并非如此。
上述代码创建了一个新的列表并将局部名称 l 绑定到它。无论 l 之前绑定到哪个对象,都不再与名称 l 相关。此后在此作用域中使用 l,都将引用这个新创建的列表。
请看下面这段单线程代码:
a = b = [1]  # a and b are both bound to the same list
print a,b    # [1] [1]
b[0] = 2     # modifies the object to which a and b are bound
print a,b    # [2] [2]
b = [0]      # now b is bound to a new list
print a,b    # [2] [0]

注意 b[0] = 2b = [0] 的区别。在第一个示例中,修改的是绑定到b的对象,而在第二个示例中,b被绑定到了一个全新的对象。

同样地,在您的代码中 l = [0] 将会把l 绑定到一个新的对象上,而您将无法重新获得对原始对象的任何引用。


好的,我明白那该怎么做。在main()的范围内,l会被绑定到thread1和thread2启动调用中绑定到da_list的值。在consum函数的范围内,写入l = [0]会将该范围内的l与prod函数范围内绑定的内存对象中的l解除绑定。 - cgiustini

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接