【GStreamer开发】GStreamer播放教程04——既看式流
目的
在《GStreamer基础教程——流》里面我们展示了如何在较差的网络条件下使用缓冲这个机制来提升用户体验。本教程在《GStreamer基础教程——流》的基础上在扩展了一下,增加了把流的内容在本地存储。并且展示了:
如何开启既看式下载
如何知道下载的是什么
如何知道在哪里下载
如何限制下载数据的总量
介绍
当播放流的时候,从网络上获得的数据被锁住之后,会创建称为future-data的一个小的缓冲区。然而,在数据播放渲染之后就会被丢弃。这就意味着,如果用户想要倒回前面去看,相应地数据仍然需要再次下载。
像YouTube一样,播放流时播放器往往会裁剪,通常会把所有下载的数据都在本地保存,还会提供一个图形化的界面来显示已经下载了多少内容。
playbin2通过DOWNLOAD标志提供了一个比较类似的功能,它会把数据在本次临时保存起来用于在播放已经下载的部分时可以保持顺畅。
代码里面同时展示了如何使用缓冲查询,它可以让你知道哪部分的文件已经可用了。
一个适应网络并在本地存储数据的例子
- <span style="font-size:14px;">#include <gst/gst.h>
- #include <string.h>
- #define GRAPH_LENGTH 80
- /* playbin2 flags */
- typedef enum {
- << 7) /* Enable progressive download (on selected formats) */
- } GstPlayFlags;
- typedef struct _CustomData {
- gboolean is_live;
- GstElement *pipeline;
- GMainLoop *loop;
- gint buffering_level;
- } CustomData;
- static void got_location (GstObject *gstobject, GstObject *prop_object, GParamSpec *prop, gpointer data) {
- gchar *location;
- g_object_get (G_OBJECT (prop_object), "temp-location", &location, NULL);
- g_print ("Temporary file: %s\n", location);
- /* Uncomment this line to keep the temporary file after the program exits */
- /* g_object_set (G_OBJECT (prop_object), "temp-remove", FALSE, NULL); */
- }
- static void cb_message (GstBus *bus, GstMessage *msg, CustomData *data) {
- switch (GST_MESSAGE_TYPE (msg)) {
- case GST_MESSAGE_ERROR: {
- GError *err;
- gchar *debug;
- gst_message_parse_error (msg, &err, &debug);
- g_print ("Error: %s\n", err->message);
- g_error_free (err);
- g_free (debug);
- gst_element_set_state (data->pipeline, GST_STATE_READY);
- g_main_loop_quit (data->loop);
- break;
- }
- case GST_MESSAGE_EOS:
- /* end-of-stream */
- gst_element_set_state (data->pipeline, GST_STATE_READY);
- g_main_loop_quit (data->loop);
- break;
- case GST_MESSAGE_BUFFERING:
- /* If the stream is live, we do not care about buffering. */
- if (data->is_live) break;
- gst_message_parse_buffering (msg, &data->buffering_level);
- /* Wait until buffering is complete before start/resume playing */
- 00)
- gst_element_set_state (data->pipeline, GST_STATE_PAUSED);
- else
- gst_element_set_state (data->pipeline, GST_STATE_PLAYING);
- break;
- case GST_MESSAGE_CLOCK_LOST:
- /* Get a new clock */
- gst_element_set_state (data->pipeline, GST_STATE_PAUSED);
- gst_element_set_state (data->pipeline, GST_STATE_PLAYING);
- break;
- default:
- /* Unhandled message */
- break;
- }
- }
- static gboolean refresh_ui (CustomData *data) {
- GstQuery *query;
- gboolean result;
- query = gst_query_new_buffering (GST_FORMAT_PERCENT);
- result = gst_element_query (data->pipeline, query);
- if (result) {
- gint n_ranges, range, i;
- gchar graph[GRAPH_LENGTH + 1];
- GstFormat format = GST_FORMAT_TIME;
- 4 position = 0, duration = 0;
- memset (graph, ' ', GRAPH_LENGTH);
- graph[GRAPH_LENGTH] = '\0';
- n_ranges = gst_query_get_n_buffering_ranges (query);
- ; range < n_ranges; range++) {
- 4 start, stop;
- gst_query_parse_nth_buffering_range (query, range, &start, &stop);
- 00;
- 00;
- for (i = (gint)start; i < stop; i++)
- graph [i] = '-';
- }
- if (gst_element_query_position (data->pipeline, &format, &position) &&
- GST_CLOCK_TIME_IS_VALID (position) &&
- gst_element_query_duration (data->pipeline, &format, &duration) &&
- GST_CLOCK_TIME_IS_VALID (duration)) {
- ));
- 00 ? 'X' : '>';
- }
- g_print ("[%s]", graph);
- 00) {
- g_print (" Buffering: %3d%%", data->buffering_level);
- } else {
- g_print (" ");
- }
- g_print ("\r");
- }
- return TRUE;
- }
- int main(int argc, charchar *argv[]) {
- GstElement *pipeline;
- GstBus *bus;
- GstStateChangeReturn ret;
- GMainLoop *main_loop;
- CustomData data;
- guint flags;
- /* Initialize GStreamer */
- gst_init (&argc, &argv);
- /* Initialize our data structure */
- , sizeof (data));
- 00;
- /* Build the pipeline */
- pipeline = gst_parse_launch ("playbin2 uri=http://docs.gstreamer.com/media/sintel_trailer-480p.webm", NULL);
- bus = gst_element_get_bus (pipeline);
- /* Set the download flag */
- g_object_get (pipeline, "flags", &flags, NULL);
- flags |= GST_PLAY_FLAG_DOWNLOAD;
- g_object_set (pipeline, "flags", flags, NULL);
- /* Uncomment this line to limit the amount of downloaded data */
- /* g_object_set (pipeline, "ring-buffer-max-size", (guint64)4000000, NULL); */
- /* Start playing */
- ret = gst_element_set_state (pipeline, GST_STATE_PLAYING);
- if (ret == GST_STATE_CHANGE_FAILURE) {
- g_printerr ("Unable to set the pipeline to the playing state.\n");
- gst_object_unref (pipeline);
- ;
- } else if (ret == GST_STATE_CHANGE_NO_PREROLL) {
- data.is_live = TRUE;
- }
- main_loop = g_main_loop_new (NULL, FALSE);
- data.loop = main_loop;
- data.pipeline = pipeline;
- gst_bus_add_signal_watch (bus);
- g_signal_connect (bus, "message", G_CALLBACK (cb_message), &data);
- g_signal_connect (pipeline, "deep-notify::temp-location", G_CALLBACK (got_location), NULL);
- /* Register a function that GLib will call every second */
- , (GSourceFunc)refresh_ui, &data);
- g_main_loop_run (main_loop);
- /* Free resources */
- g_main_loop_unref (main_loop);
- gst_object_unref (bus);
- gst_element_set_state (pipeline, GST_STATE_NULL);
- gst_object_unref (pipeline);
- g_print ("\n");
- ;
- }</span>
工作流程
这份代码是基于《GStreamer基础教程——流》里面例子的,我们仅仅看一下不同的地方即可。
创建
- <span style="font-size:14px;"> /* Set the download flag */
- g_object_get (pipeline, "flags", &flags, NULL);
- flags |= GST_PLAY_FLAG_DOWNLOAD;
- g_object_set (pipeline, "flags", flags, NULL);</span>
通过设置这个标志,playbin2通知它内部的queue存储所有下载的数据。
- <span style="font-size:14px;">g_signal_connect (pipeline, "deep-notify::temp-location", G_CALLBACK (got_location), NULL);</span>
当它们的子element属性发生变化时,playbin2就会发出deep-notify信号。在这里我们希望知道temp-location属性是什么时候变化的,了解queue2会把下载的数据存在哪里。
- <span style="font-size:14px;">static void got_location (GstObject *gstobject, GstObject *prop_object, GParamSpec *prop, gpointer data) {
- gchar *location;
- g_object_get (G_OBJECT (prop_object), "temp-location", &location, NULL);
- g_print ("Temporary file: %s\n", location);
- /* Uncomment this line to keep the temporary file after the program exits */
- /* g_object_set (G_OBJECT (prop_object), "temp-remove", FALSE, NULL); */
- }</span>
这个temp-location属性是从发出信号的element那里获得的并打印出来。
当pipeline状态从PAUSED切换到READY时,这个文件会被删除。正如注释里面写的那样,你可以通过设置queue2的temp-remove属性位FALSE来保留下载的数据。
UI
在main函数里我们启动了一个1s的定时器,这样可以每秒刷新一下UI界面。
- /* Register a function that GLib will call every second */
- , (GSourceFunc)refresh_ui, &data);
refresh_ui方法会查询pipeline来了解当前下载的数据在文件的位置以及当前播放的位置。并且用一种动画的方式在屏幕上显示出来。
- [---->------- ]
这里的'-'代表的是下载的部分,'>'代表的是当前播放的位置(当暂停的时候变成'X'位置)。当你的网络速度足够快得时候你可能会看不到下载的动画,在开始的时候就下载结束了。
- static gboolean refresh_ui (CustomData *data) {
- GstQuery *query;
- gboolean result;
- query = gst_query_new_buffering (GST_FORMAT_PERCENT);
- result = gst_element_query (data->pipeline, query);
我们在refresh_ui里面做的第一件事是就是用gst_query_new_buffering()创建一个GstQuery对象,并用gst_element_query()传给playbin2。在《GStreamer基础教程04——时间管理》里面我们展示了如何用明确的方法来查询位置/播放总时间等,如果要查询更复杂一些的内容(比如缓冲),那么我们会用更通用的gst_element_query()方法。
缓冲的查询可以基于不同的GstFormat,并非所有的element都可以响应所有格式的查询,所以需要检查在pipeline里支持哪些格式。如果gst_element_query()返回TRUE,那么查询是成功的。查询的结果用GstQuery封装起来,可以用下面的方法来解析:
- n_ranges = gst_query_get_n_buffering_ranges (query);
- ; range < n_ranges; range++) {
- 4 start, stop;
- gst_query_parse_nth_buffering_range (query, range, &start, &stop);
- 00;
- 00;
- for (i = (gint)start; i < stop; i++)
- graph [i] = '-';
- }
数据并不需要保证被按照顺序从头开始下载,因为跳跃播放时会让下载从一个新的地方开始。因此,gst_query_get_n_buffering_ranges()返回下载块的数目或者范围,然后我们用gst_query_parse_nth_buffering_rang()方法来解析下载块的位置和大小。
我们在调用gst_query_new_buffering()的请求会决定返回数据的格式,在这个例子里面,返回值是比例。这些查询到得数据用来绘制UI的下载动画。
- if (gst_element_query_position (data->pipeline, &format, &position) &&
- GST_CLOCK_TIME_IS_VALID (position) &&
- gst_element_query_duration (data->pipeline, &format, &duration) &&
- GST_CLOCK_TIME_IS_VALID (duration)) {
- ));
- 00 ? 'X' : '>';
- }
下一步就是当前位置的查询。它也支持比例的格式,所以代码和前面应该比较类似。不过这部分目前支持不是很好,所以我们使用了时间这个格式。
当前位置使用'>'或者'X'来表示,如果缓冲不到100%,cb_message会让pipeline处于PAUSE状态,那样我们就显示'X',如果已经满了100%,那么pipeline就在PLAYING状态,我们就显示'>'。
- 00) {
- g_print (" Buffering: %3d%%", data->buffering_level);
- } else {
- g_print (" ");
- }
最后,如果缓冲时钟小于100%,我们就把这个数据显示出来。
限制下载文件的大小
- /* Uncomment this line to limit the amount of downloaded data */
- /* g_object_set (pipeline, "ring-buffer-max-size", (guint64)4000000, NULL); */
打开139行的注释,让我们看看这个是如何做到的。缩小临时文件的大小,这样播放过的区域就会被覆盖。
【GStreamer开发】GStreamer播放教程04——既看式流的更多相关文章
- IE开发人员工具教程
写在前面 一直非常谷歌的控制台,因为我是做前端的,谷歌浏览器在我看来是解析JS最快的浏览器,所谓的熟能生巧,用熟悉了谷歌浏览器之后就特别喜欢用谷歌的控制台调试脚本.改变样式.查看HTML.查看资源加载 ...
- 【GStreamer开发】GStreamer基础教程13——播放速度
目标 快进,倒放和慢放是trick模式的共同技巧,它们有一个共同点就是它们都修改了播放的速度.本教程会展示如何来获得这些效果和如何进行逐帧的跳跃.主要内容是: 如何来变换播放的速度,变快或者变慢,前进 ...
- 【GStreamer开发】GStreamer基础教程04——时间管理
目标 本教程主要讲述一些和时间相关的内容.主要包括: 1. 如何问pipeline查询到流的总时间和当前播放的时间 2. 如何在流内部实现跳转功能 介绍 GstQuery是向一个element或者pa ...
- 【GStreamer开发】GStreamer播放教程01——playbin2的使用
目标 我们前面已经使用过了playbin2这个element,它可以让我们做的很少而实现很多.本教程会展示当这个element的默认设置在一些特殊情形下不符合我们的需求是可以做的一些深度定制,我们会看 ...
- 【GStreamer开发】GStreamer基础教程05——集成GUI工具
目标 本教程展示了如何在GStreamer集成一个GUI(比如:GTK+).最基本的原则是GStreamer处理多媒体的播放而GUI处理和用户的交互. 在这个教程里面,我们可以学到: 如何告诉GStr ...
- gstreamer应用开发(播放器)之旅
GStreamer开发,主要分为两块:应用开发.插件开发. 插件开发人员,通常是编解码库的作者(做出了编解码库后,希望gstreamer能用起来这个库,因此增加这个适配层).芯片原厂人员(将自家的hw ...
- 【转】一步一步教你在Ubuntu12.04搭建gstreamer开发环境
原文网址:http://blog.csdn.net/xsl1990/article/details/8333062 闲得蛋疼 无聊寂寞冷 随便写写弄弄 看到网上蛮多搭建gstreamer开 ...
- 安装gstreamer开发环境
ubuntu中安装gstreamer开发环境: * 安装gstreamer基本库,工具,以及插件 sudo apt--dev gstreamer-tools gstreamer0.-tools gst ...
- C#开发Unity游戏教程循环遍历做出判断及Unity游戏示例
C#开发Unity游戏教程循环遍历做出判断及Unity游戏示例 Unity中循环遍历每个数据,并做出判断 很多时候,游戏在玩家做出判断以后,游戏程序会遍历玩家身上大量的所需数据,然后做出判断,即首先判 ...
随机推荐
- read()和write(),读和写的优化。
读和写的优化在输入数据后输出数据十分多的情况下是十分有用的,比scanf和printf也要快. 读: int read(){ ; ; char c=getchar(); '){ if(c=='-') ...
- RS码的突发纠错能力
RS码便于纠突发错误.所谓突发错误,是指burst errors. 即一长串连续位出错.例如 0011XXXX1010. 其中X表示出错.如果是GF(2^4)中定义的RS码,则可以由一个符号错误纠正. ...
- 第02组 Alpha冲刺(3/6)
第02组 Alpha冲刺(3/6) 队名:無駄無駄组长博客作业博客 组员情况 张越洋 过去两天完成了哪些任务 摸鱼 提交记录(全组共用) 接下来的计划 沟通前后端成员,监督.提醒他们尽快完成各自的 ...
- 5.lock 锁
中断: 线程实例.interrupt(); lock锁的使用 package com.jlong; import java.util.concurrent.locks.Condition; imp ...
- [Shell]利用JS文件反弹Shell
0x01 模拟环境 攻击: kali ip: 192.168.248.132 测试: windows 7 x64 ip: 192.168.248.136 0x02 工具地址 https://githu ...
- C语言JSON序列化/反序列化
最近想找一个C语言处理嵌套结构体和结构体数组的json库,理想的是能够很容易处理复杂结构体嵌套,并且使用简单的,但是没找到比较合适的,于是打算自己封装一个: 两个问题: C语言结构体本身没有元数据,这 ...
- 深度学习面试题18:网中网结构(Network in Network)
目录 举例 参考资料 网中网结构通过多个分支的运算(卷积或池化),将分支上的运算结果在深度上连接 举例 一个3*3*2的张量, 与3个1*1*2的卷积核分别same卷积,步长=1, 与2个2*2*2的 ...
- file_put_contents 和php://input 实现存储数据进图片中
<?php /** *Recieve p_w_picpath data **/ error_reporting(E_ALL); function get_contents() { $xmlstr ...
- springboot整合vue实现上传下载文件
https://blog.csdn.net/yhhyhhyhhyhh/article/details/89888953 文章目录 springboot整合vue实现上传下载文件 1上传下载文件api文 ...
- Mysql中如何查看慢查询以及查看线程
一.MySQL数据库有几个配置选项可以帮助我们及时捕获低效SQL语句 1,slow_query_log这个参数设置为ON,可以捕获执行时间超过一定数值的SQL语句. 2,long_query_time ...