Eu tenho algumas funções computacionalmente intensivas no meu script python que gostaria de armazenar em cache. Procurei soluções no estouro de pilha e encontrei muitos links:
- /programming/4431703/python-resettable-instance-method-memoization-decorator
- https://wiki.python.org/moin/PythonDecoratorLibrary#Memoize
- http://pythonhosted.org/cachetools/
- https://pythonhosted.org/Flask-Cache/ (usei este para aplicativos de balão, mas este não é um aplicativo de balão).
No final, acabei colando isso no meu programa. Parece bastante simples - e funciona bem.
class memoized(object):
'''Decorator. Caches a function's return value each time it is called.
If called later with the same arguments, the cached value is returned
(not reevaluated).
'''
def __init__(self, func):
self.func = func
self.cache = {}
def __call__(self, *args):
if not isinstance(args, collections.Hashable):
return self.func(*args)
if args in self.cache:
return self.cache[args]
else:
value = self.func(*args)
self.cache[args] = value
return value
def __repr__(self):
'''Return the function's docstring.'''
return self.func.__doc__
def __get__(self, obj, objtype):
'''Support instance methods.'''
return functools.partial(self.__call__, obj)
No entanto, estou me perguntando se existe uma prática recomendada canônica em Python. Acho que assumi que haveria um pacote muito usado para lidar com isso e estou confuso sobre o porquê disso não existir. http://pythonhosted.org/cachetools/ é apenas na versão .6 e a sintaxe é mais complexa do que simplesmente adicionar um decorador @memoize, como em outras soluções.
@cached_property
decorador , que é mais adequado para métodos cujos valores de retorno devem permanecer os mesmos ao longo da vida da instância.Como um método de instância pode usar atributos próprios e, em particular, modificá-los, não é possível garantir a correção de um método de instância memorizado arbitrário. Além disso, sua implementação adiciona uma restrição implícita no seu objeto para ser lavável e o cache atingido, dependendo desse hash, você está limitado às classes nas quais pode usá-lo e nos campos que pode adicionar às classes com um método memorizado (ou herdado).
Por esses motivos, se você precisar memorizar uma função, é melhor projetar transformar o método de instância caro em um método estático, passando explicitamente para ele os atributos de objeto necessários como argumentos. Isso libera suas mãos do design da classe e pode melhorar o acerto do cache. Isso também ajuda a simplificar e generalizar o código de memorização. Existem implementações refinadas para este projeto, às vezes permitindo personalizar o tamanho do cache, métodos de duração / invalidação, garantindo segurança do encadeamento, persistência ...
fonte