为了账号安全,请及时绑定邮箱和手机立即绑定

Python 3.6+:嵌套多处理管理器导致 FileNotFoundError

Python 3.6+:嵌套多处理管理器导致 FileNotFoundError

烙印99 2022-03-09 21:19:49
所以我试图在一个字典上使用多处理管理器,这是我最初的尝试:from multiprocessing import Process, Managerdef task(stat):    test['z'] += 1    test['y']['Y0'] += 5if __name__ == '__main__':    test = Manager().dict({'x': {'X0': 10, 'X1': 20}, 'y': {'Y0': 0, 'Y1': 0}, 'z': 0})    p = Process(target=task, args=(test,))    p.start()    p.join()    print(test)当然,当我运行它时,输出不是我所期望的,z正确更新而y不变!这是输出:{'x': {'X0': 10, 'X1': 20}, 'y': {'Y0': 0, 'Y1': 0}, 'z': 1}然后我用谷歌搜索,在这里找到了一个解释,显然嵌套的字典也必须是Manager().dict()s 而不是普通的 python 字典(可能从 Python 3.6 开始)。所以我做了以下事情:from multiprocessing import Process, Managerdef task(stat):    test['z'] += 1    test['y']['Y0'] += 5if __name__ == '__main__':    test = Manager().dict({'x': Manager().dict({'X0': 10, 'X1': 20}), 'y': Manager().dict({'Y0': 0, 'Y1': 0}), 'z': 0})    p = Process(target=task, args=(test,))    p.start()    p.join()    print(test)    print(test['y'])但是,我得到了这个无法解释的错误,而不是它正常工作,为了清楚起见,它分为三个部分。第一部分对应于,test['y']['Y0'] += 5而第二部分是简单的print(test),最后是输出print(test['y'])我不确定为什么会这样。内部 dicts 显然是被创建的(如输出的第二部分所示)。但是由于某种原因,它们根本无法读取或写入!为什么会这样?额外:如果我通过 python 控制台(而不是脚本)运行相同的 python 代码,错误会FileNotFoundError从ConnectionRefusedError. 但是具有相同的确切回溯!
查看完整描述

1 回答

?
FFIVE

TA贡献1797条经验 获得超6个赞

随着你每次都开始一个新的经理流程,所以你真的在嵌套经理(就像标题所说的那样),这不是它应该的方式Manager()。Manager().dict()相反,您需要做的是实例化一个Manager,然后在该 manager 实例上创建字典:


from multiprocessing import Process, Manager

from multiprocessing.managers import DictProxy



def task(test):  # use parameter `test`, else you rely on forking

    test['z'] += 1

    test['y']['Y0'] += 5



if __name__ == '__main__':


    with Manager() as m:


        test = m.dict({'x': m.dict({'X0': 10, 'X1': 20}),

                       'y': m.dict({'Y0': 0, 'Y1': 0}),

                       'z': 0})


        p = Process(target=task, args=(test,))

        p.start()

        p.join()


        print(test)

        print(test['y'])


        # convert to normal dict before closing manager for persistence

        # in parent or for printing dict behind proxies

        d = {k: dict(v) if isinstance(v, DictProxy) else v

             for k, v in test.items()}


    print(d) # Manager already closed here

示例输出:


{'x': <DictProxy object, typeid 'dict' at 0x7f98cdaaa588>, 'y': <DictProxy object, typeid 'dict' at 0x7f98cda99c50>, 'z': 1}

{'Y0': 5, 'Y1': 0}

{'x': {'X0': 10, 'X1': 20}, 'y': {'Y0': 5, 'Y1': 0}, 'z': 1}


Process finished with exit code 0

Manager.Lock如果您计划从多个进程修改管理器对象,您还需要使用它。


查看完整回答
反对 回复 2022-03-09
  • 1 回答
  • 0 关注
  • 178 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信