国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Python > 正文

Django 中使用流響應處理視頻的方法

2020-02-15 22:28:45
字體:
來源:轉載
供稿:網友

起步

利用 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 StreamingHttpResponsedef hello():  yield 'Hello,'  yield 'there!'def test(request):  return StreamingHttpResponse(hello)

根據 WSGI 協議中的,當服務器調用時,應用程序對象必須返回一個可迭代的,產生零個或多個字節串。因此我們可以通過給服務器提供生成器來完成流響應的功能。

常見的使用 StreamingHttpResponse 是一些大文件的下載等,利用它還能完成斷點續傳的功能。

視頻流

使用視頻流時可以從請求頭部中獲得起始字節數。

這字段似乎是瀏覽器自動提供的,因為html代碼中,我只需要改下視頻的 src 的從靜態地址變成路由方式而已。對于響應體而言,也要提供響應體返回的塊的一個范圍:

Content-Range 分別表示了 起始字節號-終止字節號/文件總字節 ,該響應體的內容包含了文件該范圍內的內容。處理視頻流的代碼如下:

import reimport osfrom wsgiref.util import FileWrapperfrom django.http import StreamingHttpResponsedef 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 datadef 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            
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 嘉峪关市| 西峡县| 炉霍县| 彰武县| 称多县| 承德县| 松桃| 海晏县| 沁水县| 铅山县| 宜君县| 保山市| 习水县| 安庆市| 玉田县| 青田县| 铜陵市| 临西县| 秀山| 噶尔县| 桐柏县| 汕尾市| 安乡县| 贵溪市| 普兰县| 两当县| 山东省| 崇州市| 嘉义县| 镇宁| 永德县| 太谷县| 牟定县| 麟游县| 平潭县| 浮山县| 咸阳市| 眉山市| 称多县| 萨嘎县| 顺义区|