Django 中使用流响应处理视频的方法


Posted in Python onJuly 20, 2018

起步

利用 html5 的 <video> 标签可以播放:

<video width="320" height="240" controls>
 <source src="/static/video/demo.mp4" type="video/mp4">
 您的浏览器不支持Video标签。
</video>

但是这样的方式,视频中的进度条无法使用,而且以静态文件方式返回的话,后台的程序会占用大量的内存。

使用响应流的方式能很好的解决这两个问题。

StreamingHttpResponse

大多数 Django 响应使用 HttpResponse 。这意味着响应的主体内置在内存中,并以单件形式发送到 HTTP 客户端。而如果用 StreamingHttpResponse 的方式则可以以 chunks (部分块)的方式返回。一个很简单的例子就是:

from django.http import StreamingHttpResponse

def hello():
  yield 'Hello,'
  yield 'there!'

def test(request):
  return StreamingHttpResponse(hello)

根据 WSGI 协议中的,当服务器调用时,应用程序对象必须返回一个可迭代的,产生零个或多个字节串。因此我们可以通过给服务器提供生成器来完成流响应的功能。

常见的使用 StreamingHttpResponse 是一些大文件的下载等,利用它还能完成断点续传的功能。

视频流

使用视频流时可以从请求头部中获得起始字节数。

Django 中使用流响应处理视频的方法

这字段似乎是浏览器自动提供的,因为html代码中,我只需要改下视频的 src 的从静态地址变成路由方式而已。对于响应体而言,也要提供响应体返回的块的一个范围:

Django 中使用流响应处理视频的方法

Content-Range 分别表示了 起始字节号-终止字节号/文件总字节 ,该响应体的内容包含了文件该范围内的内容。处理视频流的代码如下:

import re
import os
from wsgiref.util import FileWrapper
from django.http import StreamingHttpResponse

def file_iterator(file_name, chunk_size=8192, offset=0, length=None):
  with open(file_name, "rb") as f:
    f.seek(offset, os.SEEK_SET)
    remaining = length
    while True:
      bytes_length = chunk_size if remaining is None else min(remaining, chunk_size)
      data = f.read(bytes_length)
      if not data:
        break
      if remaining:
        remaining -= len(data)
      yield data

def stream_video(request, path):
  """将视频文件以流媒体的方式响应"""
  range_header = request.META.get('HTTP_RANGE', '').strip()
  range_re = re.compile(r'bytes\s*=\s*(\d+)\s*-\s*(\d*)', re.I)
  range_match = range_re.match(range_header)
  size = os.path.getsize(path)
  content_type, encoding = mimetypes.guess_type(path)
  content_type = content_type or 'application/octet-stream'
  if range_match:
    first_byte, last_byte = range_match.groups()
    first_byte = int(first_byte) if first_byte else 0
    last_byte = first_byte + 1024 * 1024 * 8    # 8M 每片,响应体最大体积
    if last_byte >= size:
      last_byte = size - 1
    length = last_byte - first_byte + 1
    resp = StreamingHttpResponse(file_iterator(path, offset=first_byte, length=length), status=206, content_type=content_type)
    resp['Content-Length'] = str(length)
    resp['Content-Range'] = 'bytes %s-%s/%s' % (first_byte, last_byte, size)
  else:
    # 不是以视频流方式的获取时,以生成器方式返回整个文件,节省内存
    resp = StreamingHttpResponse(FileWrapper(open(path, 'rb')), content_type=content_type)
    resp['Content-Length'] = str(size)
  resp['Accept-Ranges'] = 'bytes'
  return resp

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python获取微信小程序手机号并绑定遇到的坑
Nov 19 Python
Python使用ctypes调用C/C++的方法
Jan 29 Python
python实现批量注册网站用户的示例
Feb 22 Python
python实现栅栏加解密 支持密钥加密
Mar 20 Python
使用Python控制摄像头拍照并发邮件
Apr 23 Python
ORM Django 终端打印 SQL 语句实现解析
Aug 09 Python
python多项式拟合之np.polyfit 和 np.polyld详解
Feb 18 Python
解决python ThreadPoolExecutor 线程池中的异常捕获问题
Apr 08 Python
keras中的backend.clip用法
May 22 Python
Pycharm调试程序技巧小结
Aug 08 Python
Python安装并操作redis实现流程详解
Oct 13 Python
python+requests实现接口测试的完整步骤
Oct 27 Python
Python实现手写一个类似django的web框架示例
Jul 20 #Python
python 实现求解字符串集的最长公共前缀方法
Jul 20 #Python
python实现求两个字符串的最长公共子串方法
Jul 20 #Python
Django基础知识与基本应用入门教程
Jul 20 #Python
opencv python 2D直方图的示例代码
Jul 20 #Python
Linux下python制作名片示例
Jul 20 #Python
Python爬虫的两套解析方法和四种爬虫实现过程
Jul 20 #Python
You might like
深入file_get_contents与curl函数的详解
2013/06/25 PHP
URL编码转换,escape() encodeURI() encodeURIComponent()
2006/12/27 Javascript
JQuery 确定css方框模型(盒模型Box Model)
2010/01/22 Javascript
jQuery的Ajax时无响应数据的解决方法
2010/05/25 Javascript
jquery实现弹出窗口效果的实例代码
2013/11/28 Javascript
js/jQuery简单实现选项卡功能
2014/01/02 Javascript
JS保留两位小数,多位小数的示例代码
2014/01/07 Javascript
JavaScript中创建对象和继承示例解读
2014/02/12 Javascript
JS简单计算器实例
2015/01/20 Javascript
jQuery Validate插件实现表单强大的验证功能
2015/12/18 Javascript
JavaScript中的各种操作符使用总结
2016/05/26 Javascript
jQuery简单实现仿京东分类导航层效果
2016/06/07 Javascript
Bootstrap中文本框的宽度变窄并且加入一副验证码图片的实现方法
2016/06/23 Javascript
关于vue-router路径计算问题
2017/05/10 Javascript
jQuery实现的简单在线计算器功能
2017/05/11 jQuery
基于 Bootstrap Datetimepicker 联动
2017/08/03 Javascript
nodejs超出最大的调用栈错误问题
2017/12/27 NodeJs
js中getBoundingClientRect的作用及兼容方案详解
2018/02/01 Javascript
jQuery实现DIV响应鼠标滑过由下向上展开效果示例【测试可用】
2018/04/26 jQuery
解决Layui选择全部,换页checkbox复选框重新勾选的问题方法
2018/08/14 Javascript
Layer.js实现表格溢出内容省略号显示,悬停显示全部的方法
2019/09/16 Javascript
在vue-cli3中使用axios获取本地json操作
2020/07/30 Javascript
简单介绍Python中的len()函数的使用
2015/04/07 Python
Python编程使用tkinter模块实现计算器软件完整代码示例
2017/11/29 Python
Python中出现IndentationError:unindent does not match any outer indentation level错误的解决方法
2020/04/18 Python
python脚本实现mp4中的音频提取并保存在原目录
2020/02/27 Python
PyCharm MySQL可视化Database配置过程图解
2020/06/09 Python
ziaja齐叶雅官方海外旗舰店:来自波兰的天然护肤品牌
2017/01/02 全球购物
WoolOvers爱尔兰:羊绒、羊毛和棉针织品
2017/01/04 全球购物
印尼美容产品购物网站:PerfectBeauty.id
2017/12/01 全球购物
乌克兰网上服装店:Bolf.ua
2018/10/30 全球购物
JAVA程序员自荐书
2014/01/30 职场文书
违反交通法规检讨书
2014/09/10 职场文书
2016教师学习教育法心得体会
2016/01/19 职场文书
redis的list数据类型相关命令介绍及使用
2022/01/18 Redis
MySQL事务的隔离级别详情
2022/07/15 MySQL