标准 memoize 装饰器中的缓存是否安全?
例如,假设我定义了以下装饰器:
import functools
def memoize(func):
cache = {}
@functools.wraps(func)
def memoized(*args):
result = None
if args in cache:
result = cache[args]
else:
result = func(*args)
cache[args] = result
return result
return memoized
并假设我正在尝试使用它来加速递归函数的计算,例如:
@memoize
def fib(n):
result = 1
if n > 1:
result = fib(n-1) + fib(n-2)
return result
现在我想知道计算 fib() 的两个进程是否会发生冲突?例如:
if __name__ == "__main__":
from multiprocessing import Process
p1 = Process(target=fib, args=(19,))
p2 = Process(target=fib, args=(23,))
p1.start()
p2.start()
p1.join()
p2.join()
我的第一个想法是缓存保存在 fib 的上下文中,因此它在进程之间共享,这可能导致竞争条件。但是,我认为可能发生的最坏情况是,他们都会认为,比如说,fib(17) 尚未计算,并且都会继续并行计算并一个接一个地存储相同的结果-不理想,但并不可怕,我猜。但我仍然想知道是否有办法以过程安全的方式来做到这一点。
编辑:我在 memoized() 的每个分支中添加了一个打印语句,似乎每个进程都重新计算了缓存中的所有 fib 值。或许缓存不是共享的,毕竟?如果它没有被共享,我会想如果有一种过程安全的方式来共享它(以节省更多的计算)。