如何优雅地改进Django中的模板碎片缓存详解


Posted in Python onJuly 04, 2018

前言

本文主页给大家介绍了关于如何改进Django中模板碎片缓存的相关内容,关于Django模板碎片缓存大家可以先看看这篇文章,下面话不多说了,来一起看看详细的介绍吧

起步

Django 的缓存体系提供了模板片段缓存:

{% load cache %}
{% cache 500 sidebar %}
 .. sidebar ..
{% endcache %}

但使用这个模板缓存还是需要每次都把需要的变量 sidebar 传给模板,不然当缓存过期时碎片是空白的。于是就需要的视图中获取这些数据:

def test_view(request):
 # code...
 sidebar = get_data()
 return render(reqeust, 'test_view.html', {'sidebar': sidebar})

如果这个数据获取的过程比较耗时,那么这个碎片缓存形同虚设。

低级缓存

使用低级缓存能解决数据获取耗时问题:

from django.core.cache import cache
def get_data():
 key = 'hot-course'
 result = cache.get(key)
 if result:
 return result
 # 比较耗时的数据获取
 result = Course.objects.filter().order_by('-fav_num')[:10]
 cache.set(key, result, 600) # 保存至缓存
 return result

这样一般就能解决数据来源耗时问题,一般用了这个方式就不会再用模板碎片缓存了,不然内存中就有两个缓存了,一个是原始数据,另一个是渲染成 html 代码的结果。有点多余,内存宝贵应该用于刀刃上,而且两个缓存的方式极不优雅。

使用这种底层 api 后,还是需要把数据传递到视图层,如果是公共部分的如轮播部分的视图,是会被其他模板 include 的,那就需要其他视图函数也都获取一次数据,再传递到模板层。重复的代码会很多。

有没有一种好的办法解决这种情况呢?

优雅的改进碎片缓存

改进的碎片缓存需要能按需获取,最好不需要视图层的参与。这个要求可以通过标签来实现,我们来自己实现一下这个缓存标签,在此之前呢,需要做个通用的缓存工具,能够传入数据获取的函数来做回调,这部分其实和 Django 的 django.templatetags.CacheNode 类基本一样。我这边就写与其不一样的地方:

class UserCacheNode(Node):
 """
 优雅的自定义模板碎片缓存
 """
 def __init__(self, nodelist, expire_time_var, fragment_name, vary_on, cache_name, fun=None):
 # ...
 self.fun = fun # 用于数据获取的回调函数

 def render(self, context:dict):
 # ...
 if value is None:
  if self.fun: # 实行回调
  context.update(self.fun(*vary_on))
  value = self.nodelist.render(context)
  fragment_cache.set(cache_key, value, expire_time) # 保存至缓存
 return value

然后是制作自定义标签:

def get_hot_course():
 # 做会调用,函数返回字典
 print("call hot course")
 hot_courses = Course.objects.filter().order_by('stu_nums')[:5]
 return locals()

@register.tag('hot_course_cache') # 自定义的标签名称
def hot_course_cache(parser, token):
 nodelist = parser.parse(('endcache',))
 parser.delete_first_token()
 tokens = token.split_contents()
 cache_name = None

 return UserCacheNode(
 nodelist, parser.compile_filter(tokens[1]),
 tokens[2], # fragment_name can't be a variable.
 [parser.compile_filter(t) for t in tokens[3:]],
 cache_name,
 fun=get_hot_course, # 回调函数
 )

然后在模板中就可以这么使用:

{% load course_tag %}
{% hot_course_cache 500 hot_courses %}
 ...hot_courses...
{% endcache %}

通过用自定义标签的方式,就无需视图层的参与了,缓存标签的使用方式也和体系中的 cache 相似,由于是自定义的标签,一些 IDE 会有一些警告,比如我的开发环境:

如何优雅地改进Django中的模板碎片缓存详解

运行上是没问题的,IDE 可能对这类自定义标签的支持度不是很好吧。

参考

Django's cache framework

Custom template tags and filters

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
flask中使用SQLAlchemy进行辅助开发的代码
Feb 10 Python
python中函数默认值使用注意点详解
Jun 01 Python
python中的闭包函数
Feb 09 Python
TensorFlow变量管理详解
Mar 10 Python
Python HTML解析模块HTMLParser用法分析【爬虫工具】
Apr 05 Python
Python 存储字符串时节省空间的方法
Apr 23 Python
Python基础学习之时间转换函数用法详解
Jun 18 Python
使用Python在Windows下获取USB PID&VID的方法
Jul 02 Python
python opencv捕获摄像头并显示内容的实现
Jul 11 Python
Python 内置变量和函数的查看及说明介绍
Dec 25 Python
如何基于Django实现上下文章跳转
Sep 16 Python
教你怎么用Python处理excel实现自动化办公
Apr 30 Python
Django框架多表查询实例分析
Jul 04 #Python
python 借助numpy保存数据为csv格式的实现方法
Jul 04 #Python
Python将一个CSV文件里的数据追加到另一个CSV文件的方法
Jul 04 #Python
python中csv文件的若干读写方法小结
Jul 04 #Python
Python画柱状统计图操作示例【基于matplotlib库】
Jul 04 #Python
pandas将numpy数组写入到csv的实例
Jul 04 #Python
Python实现的简单排列组合算法示例
Jul 04 #Python
You might like
php array_intersect比array_diff快(附详细的使用说明)
2011/07/03 PHP
PHP正则提取不包含指定网址的图片地址的例子
2014/04/21 PHP
Discuz批量替换帖子内容的方法(使用SQL更新数据库)
2014/06/23 PHP
php下foreach提示Warning:Invalid argument supplied for foreach()的解决方法
2014/11/11 PHP
Zend Framework入门教程之Zend_Config组件用法详解
2016/12/09 PHP
JavaScript面向对象编程
2008/03/02 Javascript
读取input:file的路径并显示本地图片的方法
2013/09/23 Javascript
js中的onchange和onpropertychange (onchange无效的解决方法)
2014/03/08 Javascript
js动态创建标签示例代码
2014/06/09 Javascript
JS的数组迭代方法
2015/02/05 Javascript
javascript中Function类型详解
2015/04/28 Javascript
js实现匹配时换色的输入提示特效代码
2015/08/17 Javascript
jQuery Ajax 加载数据时异步显示加载动画
2016/08/01 Javascript
浅谈DOM的操作以及性能优化问题-重绘重排
2017/01/08 Javascript
AngulerJS学习之按需动态加载文件
2017/02/13 Javascript
js中的DOM模拟购物车功能
2017/03/22 Javascript
nodejs个人博客开发第七步 后台登陆
2017/04/12 NodeJs
微信小程序 websocket 实现SpringMVC+Spring+Mybatis
2017/08/04 Javascript
axios进阶实践之利用最优雅的方式写ajax请求
2017/12/20 Javascript
[03:21]辉夜杯主赛事 12月25日TOP5
2015/12/26 DOTA
基于python的Tkinter实现一个简易计算器
2015/12/31 Python
Python OpenCV获取视频的方法
2018/02/28 Python
Python中的并发处理之asyncio包使用的详解
2018/04/03 Python
Python装饰器使用你可能不知道的几种姿势
2019/10/25 Python
python操作cfg配置文件方式
2019/12/22 Python
Python chardet库识别编码原理解析
2020/02/18 Python
使用pandas读取表格数据并进行单行数据拼接的详细教程
2021/03/03 Python
美国高街时尚品牌:OASAP
2016/07/24 全球购物
餐饮加盟计划书
2014/01/10 职场文书
开学典礼感言
2014/02/16 职场文书
学校欢迎标语
2014/06/18 职场文书
职工宿舍管理制度
2015/08/05 职场文书
2015年度工程师评职称工作总结
2015/10/14 职场文书
导游词之贵州织金洞
2019/10/12 职场文书
Go标准容器之Ring的使用说明
2021/05/05 Golang
vue 把二维或多维数组转一维数组
2022/04/24 Vue.js