gpt4 book ai didi

python - 如何在类属性中缓存 API 数据?

转载 作者:太空宇宙 更新时间:2023-11-04 05:29:45 24 4
gpt4 key购买 nike

我想要一个类似于下面的类,我可以在其中使用属性从 API 访问数据。不过,我希望能够将数据缓存一段时间,这样我就可以快速访问该属性而不受速率限制。实现此功能最简洁的方法是什么?

from requests import get


class GitHubUser:
def __init__(self, user):
self.user = user
print("{user} has {repos} public repos!".format(
user=user, repos=self.public_repos
))

@property
def _api_data(self):
return get(
"https://api.github.com/users/{user}".format(user=self.user)
).json()

@property
def public_repos(self):
return self._api_data["public_repos"]

最佳答案

这里有一种让它看起来整洁的方法(它可能看起来过于复杂但实际上并非如此):

from uuid import uuid4
from datetime import datetime

class cached_descriptor(object):
def __init__(self, func, timeout):
self.__doc__ = getattr(func, '__doc__')
self.func = func
self.uuid = str(uuid.uuid4())
self.timeout = timeout

def __get__(self, obj, cls):
if obj is None:
return self
if not hasattr(obj, '_cache'):
obj._cache = {}
if self.uuid not in obj._cache:
obj._cache[self.uuid] = []
data = obj._cache[self.uuid]
now = datetime.now()
if not data or (now - data[1]).total_seconds() > self.timeout:
obj._cache[self.uuid] = (self.func(obj), now)
return obj._cache[self.uuid][0]

class cached_property(object):
def __init__(self, timeout):
self.timeout = timeout

def __call__(self, func):
return cached_descriptor(func, self.timeout)

分解:

  • cached_property 是装饰器工厂,以秒为单位接受 timeout 参数
  • cached_descriptor 是一个只读描述符,将对象本身的缓存值和时间戳存储在_cache dict中,在一个随机生成的uuid下,避免多个之间的冲突缓存属性
  • 在第一次调用时,该函数将始终被调用
  • 每次下一次调用,只有超过超时时间才会调用该函数

这是一个如何工作的例子:

import time

class A(object):
def __init__(self):
self.n_f = self.n_g = 0

@cached_property(0.1)
def f(self):
self.n_f += 1
print('calling f', self.n_f)
return self.n_f

@cached_property(0.5)
def g(self):
self.n_g += 1
print('calling g', self.n_g)
return self.n_g

a = A()
print('f', a.f)
print('g', a.g)
print('f', a.f)
print('g', a.g)
print('sleep 0.2')
time.sleep(0.2)
print('f', a.f)
print('g', a.g)
print('sleep 0.4')
time.sleep(0.4)
print('f', a.f)
print('g', a.g)

哪些输出

calling f 1
f 1
calling g 1
g 1
f 1
g 1
sleep 0.2
calling f 2
f 2
g 1
sleep 0.4
calling f 3
f 3
calling g 2
g 2

关于python - 如何在类属性中缓存 API 数据?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37503506/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com