如何优雅地改进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 相关文章推荐
Python AES加密实例解析
Jan 18 Python
实用自动化运维Python脚本分享
Jun 04 Python
完美解决python中ndarray 默认用科学计数法显示的问题
Jul 14 Python
Python实现繁体中文与简体中文相互转换的方法示例
Dec 18 Python
对python多线程中互斥锁Threading.Lock的简单应用详解
Jan 11 Python
OpenCV 边缘检测
Jul 10 Python
Flask框架模板继承实现方法分析
Jul 31 Python
python命令 -u参数用法解析
Oct 24 Python
python入门之基础语法学习笔记
Feb 08 Python
基于python实现简单C/S模式代码实例
Sep 14 Python
基于Python正确读取资源文件
Sep 14 Python
Python批量将csv文件转化成xml文件的实例
May 10 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
Mysql的常用命令
2006/10/09 PHP
在windows iis5下安装php4.0+mysql之我见
2006/10/09 PHP
php中防止恶意刷新页面的代码小结
2012/10/31 PHP
PHP实现在线阅读PDF文件的方法
2015/06/17 PHP
PHP大文件分块上传功能实例详解
2019/07/22 PHP
PHP终止脚本运行三种实现方法详解
2020/09/01 PHP
给Javascript数组插入一条记录的代码
2007/08/30 Javascript
JavaScript 事件属性绑定带参数的函数
2009/03/13 Javascript
jQuery1.6 类型判断实现代码
2011/09/01 Javascript
让新消息在网页标题闪烁提示的jQuery代码
2013/11/04 Javascript
js实现的捐赠管理完整实例
2015/01/20 Javascript
JQuery中绑定事件(bind())和移除事件(unbind())
2015/02/27 Javascript
原生js编写autoComplete插件
2016/04/13 Javascript
Bootstrap Chart组件使用教程
2016/04/28 Javascript
javascript实现滚动效果的数字时钟实例
2016/07/21 Javascript
jQuery内容过滤选择器用法示例
2016/09/09 Javascript
微信小程序 获取相册照片实例详解
2016/11/16 Javascript
Angular1.x复杂指令实例详解
2017/03/01 Javascript
Node.js笔记之process模块解读
2018/05/31 Javascript
使用elementUI实现将图片上传到本地的示例
2018/09/04 Javascript
python实现的二叉树算法和kmp算法实例
2014/04/25 Python
Python如何生成树形图案
2018/01/03 Python
Python 获取中文字拼音首个字母的方法
2018/11/28 Python
树莓派3 搭建 django 服务器的实例
2019/08/29 Python
Python基于requests实现模拟上传文件
2020/04/21 Python
Django Admin后台模型列表页面如何添加自定义操作按钮
2020/11/11 Python
Python爬取酷狗MP3音频的步骤
2021/02/26 Python
百思买加拿大:Best Buy Canada
2018/03/20 全球购物
英国电子产品购物网站:TobyDeals
2018/07/30 全球购物
保卫科工作岗位职责
2014/03/01 职场文书
高中生职业规划范文
2014/03/09 职场文书
我们的节日春节活动方案
2014/08/22 职场文书
销售人才自我评价范文
2014/09/27 职场文书
结婚保证书(卖身契)
2015/02/26 职场文书
《海上日出》教学反思
2016/02/23 职场文书
《堡垒之夜》联动《刺客信条》 4月7日正式上线
2022/04/06 其他游戏