我有一个相对简单的python多处理脚本,它设置了一个工作池,通过自定义管理器将输出附加到pandas dataframe
。我发现的是当我在池上调用close()/ join()时,并非所有apply_async提交的任务都已完成。
这是一个提交1000个作业的简化示例,但只有一半完成导致断言错误。我是否忽略了一些非常简单的事情,或者这可能是一个错误?
from pandas import DataFrame
from multiprocessing.managers import BaseManager, Pool
class DataFrameResults:
def __init__(self):
self.results = DataFrame(columns=("A", "B"))
def get_count(self):
return self.results["A"].count()
def register_result(self, a, b):
self.results = self.results.append([{"A": a, "B": b}], ignore_index=True)
class MyManager(BaseManager): pass
MyManager.register('DataFrameResults', DataFrameResults)
def f1(results, a, b):
results.register_result(a, b)
def main():
manager = MyManager()
manager.start()
results = manager.DataFrameResults()
pool = Pool(processes=4)
for (i) in range(0, 1000):
pool.apply_async(f1, [results, i, i*i])
pool.close()
pool.join()
print results.get_count()
assert results.get_count() == 1000
if __name__ == "__main__":
main()
答案 0 :(得分:3)
[编辑] 您看到的问题是由于此代码:
self.results = self.results.append(...)
这不是原子的。因此,在某些情况下,线程会在阅读self.results
之后(或在追加时)被中断,但在它可以将新帧分配给self.results
- >之前这个例子将丢失。
正确的解决方案是等待使用结果对象获取结果,然后将所有结果附加到主线程中。