头条项目缓存实现
头条项目缓存实现
以用户信息数据缓存为例
common/cache/user.py
from flask import current_app from redis.exceptions import RedisError import json from sqlalchemy.orm import load_only from models.user import User from . import constants class UserProfileCache(object): """ 用户资料信息缓存 """ def __init__(self, user_id): self.key = 'user:{}:info'.format(user_id) self.user_id = user_id def save(self): """ 查询数据库保存缓存记录 :return: """ r = current_app.redis_cluster # 查询数据库 user = User.query.options(load_only(User.name, User.profile_photo, User.introduction, User.certificate)).filter_by(id=self.user_id).first() # 判断结果是否存在 # 保存到redis中 if user is None: try: r.setex(self.key, constants.USER_NOT_EXISTS_CACHE_TTL, -1) except RedisError as e: current_app.logger.error(e) return None else: cache_data = { 'name': user.name, 'photo': user.profile_photo, 'intro': user.introduction, 'certi': user.certificate } try: r.setex(self.key, constants.UserProfileCacheTTL.get_val(), json.dumps(cache_data)) except RedisError as e: current_app.logger.error(e) return cache_data def get(self): """ 获取用户的缓存数据 :return: """ r = current_app.redis_cluster # 先查询redis try: ret = r.get(self.key) except RedisError as e: current_app.logger.error(e) ret = None if ret is not None: # 如果存在记录,读取 if ret == b'-1': # 判断记录值,如果为-1,表示用户不存在 return None # 如果不为-1,需要json转换,返回 else: return json.loads(ret) else: # 如果记录不存在, cache_data = self.save() return cache_data def clear(self): """ 清除用户缓存 """ try: current_app.redis_cluster.delete(self.key) except RedisError as e: current_app.logger.error(e) def exists(self): """ 判断用户是否存在 """ # 查询redis r = current_app.redis_cluster try: ret = r.get(self.key) except RedisError as e: current_app.logger.error(e) ret = None # 如果缓存记录存在 if ret is not None: if ret == b'-1': # 如果缓存记录为-1 ,表示用户不存在 return False else: # 如果缓存记录不为-1, 表示用户存在 return True # 如果缓存记录不存在,查询数据库 else: cache_data = self.save() if cache_data is not None: return True else: return False
common/cache/constants.py
class BaseCacheTTL(object): """ 缓存有效期 为防止缓存雪崩,在设置缓存有效期时采用设置不同有效期的方案 通过增加随机值实现 """ TTL = 0 # 由子类设置 MAX_DELTA = 10 * 60 # 随机的增量上限 @classmethod def get_val(cls): return cls.TTL + random.randrange(0, cls.MAX_DELTA) class UserProfileCacheTTL(BaseCacheTTL): """ 用户资料数据缓存时间, 秒 """ TTL = 30 * 60
接口示例
定义获取当前用户信息的接口
GET /v1_0/user
返回JSON
在toutiao/resources/user/__init__.py中定义路由
user_api.add_resource(profile.CurrentUserResource, '/v1_0/user', endpoint='CurrentUser')
在toutiao/resources/ user/profile.py 中
class CurrentUserResource(Resource): """ 用户自己的数据 """ method_decorators = [login_required] def get(self): """ 获取当前用户自己的数据 """ user_data = cache_user.UserProfileCache(g.user_id).get() user_data['id'] = g.user_id return user_data
壬戌之秋,七月既望,苏子与客泛舟游于赤壁之下。清风徐来,水波不兴。举酒属客,诵明月之诗,歌窈窕之章。少焉,月出于东山之上,徘徊于斗牛之间。白露横江,水光接天。纵一苇之所如,凌万顷之茫然。浩浩乎如冯虚御风,而不知其所止;飘飘乎如遗世独立,羽化而登仙。
于是饮酒乐甚,扣舷而歌之。歌曰:“桂棹兮兰桨,击空明兮溯流光。渺渺兮予怀,望美人兮天一方。”客有吹洞箫者,倚歌而和之。其声呜呜然,如怨如慕,如泣如诉,余音袅袅,不绝如缕。舞幽壑之潜蛟,泣孤舟之嫠妇。
苏子愀然,正襟危坐而问客曰:“何为其然也?”客曰:“月明星稀,乌鹊南飞,此非曹孟德之诗乎?西望夏口,东望武昌,山川相缪,郁乎苍苍,此非孟德之困于周郎者乎?方其破荆州,下江陵,顺流而东也,舳舻千里,旌旗蔽空,酾酒临江,横槊赋诗,固一世之雄也,而今安在哉?况吾与子渔樵于江渚之上,侣鱼虾而友麋鹿,驾一叶之扁舟,举匏樽以相属。寄蜉蝣于天地,渺沧海之一粟。哀吾生之须臾,羡长江之无穷。挟飞仙以遨游,抱明月而长终。知不可乎骤得,托遗响于悲风。”
苏子曰:“客亦知夫水与月乎?逝者如斯,而未尝往也;盈虚者如彼,而卒莫消长也。盖将自其变者而观之,则天地曾不能以一瞬;自其不变者而观之,则物与我皆无尽也,而又何羡乎!且夫天地之间,物各有主,苟非吾之所有,虽一毫而莫取。惟江上之清风,与山间之明月,耳得之而为声,目遇之而成色,取之无禁,用之不竭,是造物者之无尽藏也,而吾与子之所共适。”
客喜而笑,洗盏更酌。肴核既尽,杯盘狼籍。相与枕藉乎舟中,不知东方之既白。

浙公网安备 33010602011771号