知道是否已经访问了python缓存属性,而没有实际访问它

2024-04-25 23:02:44 发布

您现在位置:Python中文网/ 问答频道 /正文

有许多方法装饰器的示例,它们将方法转换为缓存属性。但有时,我想检查缓存是否处于“活动”状态,这意味着该属性已被访问,缓存已被填充

例如,如果我使用rows缓存在rows中存储一个sql表,我希望根据缓存计算表的长度(如果已填充),如果未填充,则通过单独的sql调用。如何在不触发其访问的情况下检查rows是否已被访问

这是一个很好的装饰器,取自David Beazley的“Python食谱”),我正在使用它满足缓存属性的需要。我已经对它进行了增强,以支持我当前的攻击

class lazyprop:
    def __init__(self, func):
        self.func = func

    def __get__(self, instance, cls):
        if instance is None:
            return self
        else:
            value = self.func(instance)
            setattr(instance, self.func.__name__, value)
            setattr(instance, self.func.__name__ + '__cache_active', True)  # my hack
            return value

使用示例:

>>> class Test:
...     def __init__(self, a):
...         self.a = a
...     @lazyprop
...     def len(self):
...         print('generating "len"')
...         return len(self.a)
>>> t = Test([0, 1, 2])
>>> # See what happens if I ask if there is a 'len' attribute:
>>> hasattr(t, 'len')
generating "len"
3
>>> t.len
5

所以hasattr实际上触发了len方法调用,所以我不能使用它。无论如何,我不想使用它,因为我不是要求属性(键/引用)的存在,而是要求它的值的存在(即之前的计算)

根据标有“我的黑客”的行,我现在可以这样做:

def has_active_cache(instance, attr):
    return getattr(instance, attr + '__cache_active', False)
>>> t = Test([0, 1, 2])
>>> print("Accessed:", has_active_cache(t, 'len'))
Accessed: False
>>> t.len
generating "len"
3
>>> print("Accessed:", has_active_cache(t, 'len'))
Accessed: True

但我相信还有比这更优雅的解决方案。也许一个会和lazyprop本身“结合”在一起


Tags: 方法instanceselfcachelenreturnif属性
1条回答
网友
1楼 · 发布于 2024-04-25 23:02:44

仅供参考,通过functools,属性缓存是Python 3.8标准库的一部分

https://docs.python.org/3/library/functools.html?highlight=s#functools.cached_property

使用这个装饰器,您可以直接访问类的__dict__属性来检查值是否被缓存

使用文档中的示例

import statistics
from functools import cached_property


class DataSet:
    def __init__(self, sequence_of_numbers):
        self._data = sequence_of_numbers

    @cached_property
    def stdev(self):
        return statistics.stdev(self._data)

    @cached_property
    def variance(self):
        return statistics.variance(self._data)

然后测试它

ds = DataSet(range(1, 20))
ds.stdev
5.627314338711377
ds.__dict__
{'_data': range(1, 20), 'stdev': 5.627314338711377}
ds.variance
31.666666666666668
ds.__dict__
{'_data': range(1, 20), 'stdev': 5.627314338711377, 'variance': 31.666666666666668}

相关问题 更多 >

    热门问题