libvlc视频播放器: 补偿播放进度时间-更加精细

简介: 在上几个章节,介绍了如何使用libvlc开发一款视频播放器,libvlc除了具备普通播放器该有的功能,还有很多其他强大的功能,比如:播放rtsp\rtmp流媒体视频、录制画面为视频、各种滤镜等等;

一、问题描述与解决思路

在上几个章节,介绍了如何使用libvlc开发一款视频播放器,libvlc除了具备普通播放器该有的功能,还有很多其他强大的功能,比如:播放rtsp\rtmp流媒体视频、录制画面为视频、各种滤镜等等;

在开发视频播放器的过程中,肯定是需要显示视频的事实播放进度的,这个功能在上篇文章里已经介绍了,采用libvlc的回调事件来获取当前视频的播放进度。但是这个回调事件里,得到的进度不够精细,固定一秒只能返回2次左右,也就是500ms一次,靠回调返回的时间直接给进度条控件赋值,会感觉不细腻,进度条一跳一跳的,不细腻。 libvlc本身接口上没有提供修改时间间隔精度的功能,这里采用的方法是: 采用系统时间补偿的思路,计算当前播放的位置。

下面这个是libvlc事件里打印的时间:

pos: 58152
pos: 58652
pos: 59151
pos: 59402
pos: 59900
pos: 60151
pos: 60401
pos: 60902
pos: 61154
pos: 61651
.......

可以看到,时间的间隔差不多是500ms返回一次。

这是当前正在播放视频的媒体详细信息:

{
    "streams": [
        {
            "index": 0,
            "codec_name": "h264",
            "codec_long_name": "H.264 / AVC / MPEG-4 AVC / MPEG-4 part 10",
            "profile": "High",
            "codec_type": "video",
            "codec_time_base": "125/5994",
            "codec_tag_string": "avc1",
            "codec_tag": "0x31637661",
            "width": 1280,
            "height": 720,
            "coded_width": 1280,
            "coded_height": 720,
            "has_b_frames": 2,
            "sample_aspect_ratio": "1:1",
            "display_aspect_ratio": "16:9",
            "pix_fmt": "yuv420p",
            "level": 41,
            "chroma_location": "left",
            "refs": 1,
            "is_avc": "true",
            "nal_length_size": "4",
            "r_frame_rate": "2997/125",
            "avg_frame_rate": "2997/125",
            "time_base": "1/11988",
            "start_pts": 0,
            "start_time": "0.000000",
            "duration_ts": 67452500,
            "duration": "5626.668335",
            "bit_rate": "1498216",
            "bits_per_raw_sample": "8",
            "nb_frames": "134905",
            "disposition": {
                "default": 1,
                "dub": 0,
                "original": 0,
                "comment": 0,
                "lyrics": 0,
                "karaoke": 0,
                "forced": 0,
                "hearing_impaired": 0,
                "visual_impaired": 0,
                "clean_effects": 0,
                "attached_pic": 0,
                "timed_thumbnails": 0
            },
            "tags": {
                "creation_time": "2018-01-25T18:44:17.000000Z",
                "language": "und",
                "handler_name": "VideoHandler"
            }
        },
        {
            "index": 1,
            "codec_name": "aac",
            "codec_long_name": "AAC (Advanced Audio Coding)",
            "profile": "LC",
            "codec_type": "audio",
            "codec_time_base": "1/48000",
            "codec_tag_string": "mp4a",
            "codec_tag": "0x6134706d",
            "sample_fmt": "fltp",
            "sample_rate": "48000",
            "channels": 6,
            "channel_layout": "5.1",
            "bits_per_sample": 0,
            "r_frame_rate": "0/0",
            "avg_frame_rate": "0/0",
            "time_base": "1/48000",
            "start_pts": 0,
            "start_time": "0.000000",
            "duration_ts": 270081024,
            "duration": "5626.688000",
            "bit_rate": "224000",
            "max_bit_rate": "224000",
            "nb_frames": "263753",
            "disposition": {
                "default": 1,
                "dub": 0,
                "original": 0,
                "comment": 0,
                "lyrics": 0,
                "karaoke": 0,
                "forced": 0,
                "hearing_impaired": 0,
                "visual_impaired": 0,
                "clean_effects": 0,
                "attached_pic": 0,
                "timed_thumbnails": 0
            },
            "tags": {
                "creation_time": "2018-01-25T18:44:17.000000Z",
                "language": "eng",
                "handler_name": "SoundHandler"
            }
        }
    ]
}

解决办法:
采用系统时间计算当前播放时间:

/*获取当前的播放时间*/
qint64 Widget::libvlc_GetCurrentPlayerPos()
{
    static qint64 lastPlayTime = 0;
    static qint64 lastPlayTimeMSecs = 0;
    if(vlc_mediaPlayer==nullptr)
    {
        return 0;
    }
    qint64 currentTime = libvlc_media_player_get_time(vlc_mediaPlayer);
    if (lastPlayTime == currentTime && lastPlayTime != 0)
    {
        currentTime += QDateTime::currentMSecsSinceEpoch() - lastPlayTimeMSecs;
    }
    else
    {
        lastPlayTime = currentTime;
        lastPlayTimeMSecs = QDateTime::currentMSecsSinceEpoch();
    }
    return currentTime;
}

开启一个定时器,定时调用该函数获取当前播放器的时间,想要多精确就看定时器多久获取一次了。我这里定时50ms获取一次时间,打印的效果如下:

pos: 1088
pos: 1155
pos: 1218
pos: 1281
pos: 1342
pos: 1418
pos: 1480
pos: 1544
pos: 1604
pos: 1656
pos: 1719
pos: 1782
pos: 1842
pos: 1905
pos: 1966
pos: 2029
pos: 2091
pos: 2152
pos: 2213
pos: 2273
pos: 2337
pos: 2398
pos: 2460
pos: 2521
pos: 2582
pos: 2651
pos: 2712
pos: 2789
pos: 2850
pos: 2901
pos: 2961
pos: 3019
pos: 3083
pos: 3154
pos: 3212
pos: 3281
pos: 3344
pos: 3403
pos: 3463
pos: 3535
pos: 3598
pos: 3651
pos: 3712
pos: 3773

50ms更新一次,进度条就很平滑了。

二、完整的代码

下面就贴出了和本章节主要解决的问题的相关代码。

2.1 widge.h

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <vlc/vlc.h>
#include <QDebug>
#include <QFileDialog>
#include <QMoveEvent>
#include "form.h"
#include <QTime>
#include <QThread>
#include <QTimer>

QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE

class Widget : public QWidget
{
    Q_OBJECT

public:
    Widget(QWidget *parent = nullptr);
    ~Widget();
    qint64 libvlc_GetCurrentPlayerPos();
    ........
private slots:
    void timer_update();
    ........
protected:
    ........
private:
    ........
    QTimer *timer;

};
#endif // WIDGET_H

2.2 widget.cpp

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ................
    
    //定时器更新
    timer = new QTimer(this);
    connect(timer, SIGNAL(timeout()), this, SLOT(timer_update()));
    timer->start(50);

}

/*获取当前的播放时间*/
qint64 Widget::libvlc_GetCurrentPlayerPos()
{
    static qint64 lastPlayTime = 0;
    static qint64 lastPlayTimeMSecs = 0;
    if(vlc_mediaPlayer==nullptr)
    {
        return 0;
    }
    qint64 currentTime = libvlc_media_player_get_time(vlc_mediaPlayer);
    if (lastPlayTime == currentTime && lastPlayTime != 0)
    {
        currentTime += QDateTime::currentMSecsSinceEpoch() - lastPlayTimeMSecs;
    }
    else
    {
        lastPlayTime = currentTime;
        lastPlayTimeMSecs = QDateTime::currentMSecsSinceEpoch();
    }
    return currentTime;
}


//播放器的播放进度时间更新
void Widget::timer_update()
{
    static qint64 pos1=0;
    if(vlc_mediaPlayer)
    {
        //处于播放状态
        if(libvlc_media_player_is_playing(vlc_mediaPlayer))
        {
            qint64 pos=libvlc_GetCurrentPlayerPos();
            if(pos1!=pos)
            {
                pos1=pos;
                qDebug()<<"pos:"<<pos;
                ui->label_t1->setText(QString("%1").arg(QTime(0, 0, 0,0).addMSecs(int(pos)).toString(QString::fromLatin1("HH:mm:ss:zzz"))));
                ui->horizontalSlider_pos->setValue(pos);
            }
        }
    }
}
目录
相关文章
|
3月前
|
数据处理 索引
基于Qt和OpenGL的雷达显示实例
基于Qt和OpenGL的雷达显示实例
109 0
|
开发工具 Android开发 图形学
【2015~2024】大牛直播SDK演化史
大牛直播SDK始创于2015年,旨在提供低延迟的RTMP推拉流解决方案,特别适用于毫秒级延迟的应用场景。初始版本针对Android平台实现RTMP直播推送,随后扩展至Windows、iOS和Linux平台,支持RTMP与RTSP播放器,延迟稳定在一秒以内。SDK涵盖实时推流、播放、GB28181设备接入、录像、多路流媒体转发、轻量级RTSP服务等多种功能,并支持H.265编码格式。该SDK适用于在线教育、智慧安防等多个行业场景,并已发展成为GitHub上获得超过10000星标的强大跨平台流媒体内核直播SDK。
599 1
|
12月前
|
数据采集 Java Python
如何用Python同时抓取多个网页:深入ThreadPoolExecutor
在信息化时代,实时数据的获取对体育赛事爱好者、数据分析师和投注行业至关重要。本文介绍了如何使用Python的`ThreadPoolExecutor`结合代理IP和请求头设置,高效稳定地抓取五大足球联赛的实时比赛信息。通过多线程并发处理,解决了抓取效率低、请求限制等问题,提供了详细的代码示例和解析方法。
293 0
如何用Python同时抓取多个网页:深入ThreadPoolExecutor
|
存储 数据管理 UED
在钉钉的审批系统中,审批单的状态主要有两种:"NEW"和"RUNNING"。
【2月更文挑战第19天】在钉钉的审批系统中,审批单的状态主要有两种:"NEW"和"RUNNING"。
479 5
|
存储 NoSQL 前端开发
前端轻量级数据库mongodb
【10月更文挑战第2天】MongoDB 是一个基于分布式文件存储的开源数据库系统,不属于前端轻量级数据库,而是后端数据库。它使用 BSON 格式存储数据,支持复杂的数据结构,适用于内容管理系统、物联网等领域。MongoDB 通过动态模式和面向对象的数据存储方式,提供了灵活的数据模型。在 Web 应用中,它通常作为后端存储,通过 API 与前端交互,实现高效的数据管理和实时更新。
221 3
|
前端开发 开发者
SCSS中的结构化伪类选择器详解与示例
SCSS中的结构化伪类选择器详解与示例
376 1
|
编解码 开发工具 Android开发
iOS平台如何实现毫秒级延迟的RTMP|RTSP播放器
在我的blog里面,最近很少有提到iOS平台RTMP推送|轻量级RTSP服务和RTMP|RTSP直播播放模块,实际上,我们在2016年就发布了iOS平台直播推拉流、转发模块,只是因为传统行业,对iOS的需求比较少,所以一直没单独说明,本文主要介绍下,如何在iOS平台播放RTMP或RTSP流。
245 6
解决:java.lang.UnsatisfiedLinkError: dalvik.system.PathClassLoader[DexPathList[[zip file
解决:java.lang.UnsatisfiedLinkError: dalvik.system.PathClassLoader[DexPathList[[zip file
|
机器学习/深度学习 搜索推荐 数据可视化
大数据用户画像之基本概念
大数据用户画像利用大数据技术分析用户基本信息、消费行为、兴趣、社交及地理数据,创建详细用户模型,助力企业精准营销。涉及技术包括数据挖掘、大数据处理(Hadoop、Spark)、数据可视化、机器学习和数据库管理。通过用户画像,企业可实现市场定位、个性化推荐、精准广告、产品优化和风险控制。学习该领域需掌握多个技术栈,包括相关算法、工具及业务理解。
1694 4
|
数据处理 Python
使用Pandas解决问题:对比两列数据取最大值的五种方法
​在数据处理和分析中,经常需要比较两个或多个列的值,并取其中的最大值。Pandas库作为Python中数据处理和分析的强大工具,提供了多种灵活的方法来实现这一需求。本文将详细介绍五种使用Pandas对比两列数据并取最大值的方法,通过代码示例和案例分析,帮助新手更好地理解并掌握这些技巧。
574 0