Qt-FFmpeg开发-视频播放【软解码】

更多精彩内容
个人内容分类汇总
音视频开发

1、概述

介四里沒有挽过的船新版本,挤需感受三番钟,里造会干我一样,爱象节个版本

  • 最近研究了一下FFmpeg开发,功能实在是太强大了,网上ffmpeg3、4的文章还是很多的,但是学习嘛,最新的还是不能放过,就选了一个最新的ffmpeg n5.1.2版本,和3、4版本api变化还是挺大的;
  • 在这个Demo里主要使用Qt + FFmpeg开发一个简单的视频播放器,这里使用的是软解码,硬解码后续再讲;
  • 同时为了尽可能的简单,这里没有进行音频解码和播放,只是单独的进行视频解码播放;
  • 视频显示没有上来就OpenGL、SDL、D3D,这对于初学者不太友好,所以这里使用了QPainter进行绘制,所以CPU占用还是挺高的,后面换成OpenGL就好了;

开发环境说明

  • 系统:Windows10、Ubuntu20.04
  • Qt版本:V5.12.5
  • 编译器:MSVC2017-64、GCC/G++64
  • FFmpeg版本:n5.1.2

2、实现效果

  1. 使用ffmpeg音视频库【软解码】实现的视频播放器;
  2. 支持打开本地视频文件(如mp4、mov、avi等)、网络视频流(rtsp、rtmp、http等);
  3. 支持视频【匀速播放】;
  4. 采用QPainter进行显示,支持【自适应】窗口缩放;
  5. 视频播放支持实时【开始/关闭、暂停/继续】播放;
  6. 视频解码、线程控制、显示各部分功能分离,【低耦合度】。
  7. 采用最新的【5.1.2版本】ffmpeg库进行开发,【超详细注释信息】,将所有踩过的坑、解决办法、注意事项都得很写清楚。

3、FFmpeg软解码流程

4、主要代码

  • 啥也不说了,直接上代码,一切有注释

  • videodecode.h文件

    /******************************************************************************
    * @文件名 videodecode.h
    * @功能 视频解码类,在这个类中调用ffmpeg打开视频进行解码
    *
    * @开发者 mhf
    * @邮箱 1603291350@qq.com
    * @时间 2022/09/15
    * @备注
    *****************************************************************************/
    #ifndef VIDEODECODE_H
    #define VIDEODECODE_H #include <QString>
    #include <QSize> struct AVFormatContext;
    struct AVCodecContext;
    struct AVRational;
    struct AVPacket;
    struct AVFrame;
    struct SwsContext;
    struct AVBufferRef;
    class QImage; class VideoDecode
    {
    public:
    VideoDecode();
    ~VideoDecode(); bool open(const QString& url = QString()); // 打开媒体文件,或者流媒体rtmp、strp、http
    QImage read(); // 读取视频图像
    void close(); // 关闭
    bool isEnd(); // 是否读取完成
    const qint64& pts(); // 获取当前帧显示时间 private:
    void initFFmpeg(); // 初始化ffmpeg库(整个程序中只需加载一次)
    void showError(int err); // 显示ffmpeg执行错误时的错误信息
    qreal rationalToDouble(AVRational* rational); // 将AVRational转换为double
    void clear(); // 清空读取缓冲
    void free(); // 释放 private:
    AVFormatContext* m_formatContext = nullptr; // 解封装上下文
    AVCodecContext* m_codecContext = nullptr; // 解码器上下文
    SwsContext* m_swsContext = nullptr; // 图像转换上下文
    AVPacket* m_packet = nullptr; // 数据包
    AVFrame* m_frame = nullptr; // 解码后的视频帧
    int m_videoIndex = 0; // 视频流索引
    qint64 m_totalTime = 0; // 视频总时长
    qint64 m_totalFrames = 0; // 视频总帧数
    qint64 m_obtainFrames = 0; // 视频当前获取到的帧数
    qint64 m_pts = 0; // 图像帧的显示时间
    qreal m_frameRate = 0; // 视频帧率
    QSize m_size; // 视频分辨率大小
    char* m_error = nullptr; // 保存异常信息
    bool m_end = false; // 视频读取完成
    uchar* m_buffer = nullptr; // YUV图像需要转换位RGBA图像,这里保存转换后的图形数据
    }; #endif // VIDEODECODE_H
  • videodecode.cpp文件

    #include "videodecode.h"
    #include <QDebug>
    #include <QImage>
    #include <QMutex>
    #include <qdatetime.h> extern "C" { // 用C规则编译指定的代码
    #include "libavcodec/avcodec.h"
    #include "libavformat/avformat.h"
    #include "libavutil/avutil.h"
    #include "libswscale/swscale.h"
    #include "libavutil/imgutils.h" } #define ERROR_LEN 1024 // 异常信息数组长度
    #define PRINT_LOG 1 VideoDecode::VideoDecode()
    {
    // initFFmpeg(); // 5.1.2版本不需要调用了 m_error = new char[ERROR_LEN];
    } VideoDecode::~VideoDecode()
    {
    close();
    } /**
    * @brief 初始化ffmpeg库(整个程序中只需加载一次)
    * 旧版本的ffmpeg需要注册各种文件格式、解复用器、对网络库进行全局初始化。
    * 在新版本的ffmpeg中纷纷弃用了,不需要注册了
    */
    void VideoDecode::initFFmpeg()
    {
    static bool isFirst = true;
    static QMutex mutex;
    QMutexLocker locker(&mutex);
    if(isFirst)
    {
    // av_register_all(); // 已经从源码中删除
    /**
    * 初始化网络库,用于打开网络流媒体,此函数仅用于解决旧GnuTLS或OpenSSL库的线程安全问题。
    * 一旦删除对旧GnuTLS和OpenSSL库的支持,此函数将被弃用,并且此函数不再有任何用途。
    */
    avformat_network_init();
    isFirst = false;
    }
    } /**
    * @brief 打开媒体文件,或者流媒体,例如rtmp、strp、http
    * @param url 视频地址
    * @return true:成功 false:失败
    */
    bool VideoDecode::open(const QString &url)
    {
    if(url.isNull()) return false; AVDictionary* dict = nullptr;
    av_dict_set(&dict, "rtsp_transport", "tcp", 0); // 设置rtsp流使用tcp打开,如果打开失败错误信息为【Error number -135 occurred】可以切换(UDP、tcp、udp_multicast、http),比如vlc推流就需要使用udp打开
    av_dict_set(&dict, "max_delay", "3", 0); // 设置最大复用或解复用延迟(以微秒为单位)。当通过【UDP】 接收数据时,解复用器尝试重新排序接收到的数据包(因为它们可能无序到达,或者数据包可能完全丢失)。这可以通过将最大解复用延迟设置为零(通过max_delayAVFormatContext 字段)来禁用。
    av_dict_set(&dict, "timeout", "1000000", 0); // 以微秒为单位设置套接字 TCP I/O 超时,如果等待时间过短,也可能会还没连接就返回了。 // 打开输入流并返回解封装上下文
    int ret = avformat_open_input(&m_formatContext, // 返回解封装上下文
    url.toStdString().data(), // 打开视频地址
    nullptr, // 如果非null,此参数强制使用特定的输入格式。自动选择解封装器(文件格式)
    &dict); // 参数设置
    // 释放参数字典
    if(dict)
    {
    av_dict_free(&dict);
    }
    // 打开视频失败
    if(ret < 0)
    {
    showError(ret);
    free();
    return false;
    } // 读取媒体文件的数据包以获取流信息。
    ret = avformat_find_stream_info(m_formatContext, nullptr);
    if(ret < 0)
    {
    showError(ret);
    free();
    return false;
    }
    m_totalTime = m_formatContext->duration / (AV_TIME_BASE / 1000); // 计算视频总时长(毫秒)
    #if PRINT_LOG
    qDebug() << QString("视频总时长:%1 ms,[%2]").arg(m_totalTime).arg(QTime::fromMSecsSinceStartOfDay(int(m_totalTime)).toString("HH:mm:ss zzz"));
    #endif // 通过AVMediaType枚举查询视频流ID(也可以通过遍历查找),最后一个参数无用
    m_videoIndex = av_find_best_stream(m_formatContext, AVMEDIA_TYPE_VIDEO, -1, -1, nullptr, 0);
    if(m_videoIndex < 0)
    {
    showError(m_videoIndex);
    free();
    return false;
    } AVStream* videoStream = m_formatContext->streams[m_videoIndex]; // 通过查询到的索引获取视频流 // 获取视频图像分辨率(AVStream中的AVCodecContext在新版本中弃用,改为使用AVCodecParameters)
    m_size.setWidth(videoStream->codecpar->width);
    m_size.setHeight(videoStream->codecpar->height);
    m_frameRate = rationalToDouble(&videoStream->avg_frame_rate); // 视频帧率 // 通过解码器ID获取视频解码器(新版本返回值必须使用const)
    const AVCodec* codec = avcodec_find_decoder(videoStream->codecpar->codec_id);
    m_totalFrames = videoStream->nb_frames; #if PRINT_LOG
    qDebug() << QString("分辨率:[w:%1,h:%2] 帧率:%3 总帧数:%4 解码器:%5")
    .arg(m_size.width()).arg(m_size.height()).arg(m_frameRate).arg(m_totalFrames).arg(codec->name);
    #endif // 分配AVCodecContext并将其字段设置为默认值。
    m_codecContext = avcodec_alloc_context3(codec);
    if(!m_codecContext)
    {
    #if PRINT_LOG
    qWarning() << "创建视频解码器上下文失败!";
    #endif
    free();
    return false;
    } // 使用视频流的codecpar为解码器上下文赋值
    ret = avcodec_parameters_to_context(m_codecContext, videoStream->codecpar);
    if(ret < 0)
    {
    showError(ret);
    free();
    return false;
    } m_codecContext->flags2 |= AV_CODEC_FLAG2_FAST; // 允许不符合规范的加速技巧。
    m_codecContext->thread_count = 8; // 使用8线程解码 // 初始化解码器上下文,如果之前avcodec_alloc_context3传入了解码器,这里设置NULL就可以
    ret = avcodec_open2(m_codecContext, nullptr, nullptr);
    if(ret < 0)
    {
    showError(ret);
    free();
    return false;
    } // 分配AVPacket并将其字段设置为默认值。
    m_packet = av_packet_alloc();
    if(!m_packet)
    {
    #if PRINT_LOG
    qWarning() << "av_packet_alloc() Error!";
    #endif
    free();
    return false;
    }
    // 分配AVFrame并将其字段设置为默认值。
    m_frame = av_frame_alloc();
    if(!m_frame)
    {
    #if PRINT_LOG
    qWarning() << "av_frame_alloc() Error!";
    #endif
    free();
    return false;
    } // 分配图像空间
    int size = av_image_get_buffer_size(AV_PIX_FMT_RGBA, m_size.width(), m_size.height(), 4);
    /**
    * 【注意:】这里可以多分配一些,否则如果只是安装size分配,大部分视频图像数据拷贝没有问题,
    * 但是少部分视频图像在使用sws_scale()拷贝时会超出数组长度,在使用使用msvc debug模式时delete[] m_buffer会报错(HEAP CORRUPTION DETECTED: after Normal block(#32215) at 0x000001AC442830370.CRT delected that the application wrote to memory after end of heap buffer)
    * 特别是这个视频流http://vfx.mtime.cn/Video/2019/02/04/mp4/190204084208765161.mp4
    */
    m_buffer = new uchar[size + 1000]; // 这里多分配1000个字节就基本不会出现拷贝超出的情况了,反正不缺这点内存
    // m_image = new QImage(m_buffer, m_size.width(), m_size.height(), QImage::Format_RGBA8888); // 这种方式分配内存大部分情况下也可以,但是因为存在拷贝超出数组的情况,delete时也会报错
    m_end = false;
    return true;
    } /**
    * @brief
    * @return
    */
    QImage VideoDecode::read()
    {
    // 如果没有打开则返回
    if(!m_formatContext)
    {
    return QImage();
    } // 读取下一帧数据
    int readRet = av_read_frame(m_formatContext, m_packet);
    if(readRet < 0)
    {
    avcodec_send_packet(m_codecContext, m_packet); // 读取完成后向解码器中传如空AVPacket,否则无法读取出最后几帧
    }
    else
    {
    if(m_packet->stream_index == m_videoIndex) // 如果是图像数据则进行解码
    {
    // 计算当前帧时间(毫秒)
    #if 1 // 方法一:适用于所有场景,但是存在一定误差
    m_packet->pts = qRound64(m_packet->pts * (1000 * rationalToDouble(&m_formatContext->streams[m_videoIndex]->time_base)));
    m_packet->dts = qRound64(m_packet->dts * (1000 * rationalToDouble(&m_formatContext->streams[m_videoIndex]->time_base)));
    #else // 方法二:适用于播放本地视频文件,计算每一帧时间较准,但是由于网络视频流无法获取总帧数,所以无法适用
    m_obtainFrames++;
    m_packet->pts = qRound64(m_obtainFrames * (qreal(m_totalTime) / m_totalFrames));
    #endif
    // 将读取到的原始数据包传入解码器
    int ret = avcodec_send_packet(m_codecContext, m_packet);
    if(ret < 0)
    {
    showError(ret);
    }
    }
    }
    av_packet_unref(m_packet); // 释放数据包,引用计数-1,为0时释放空间 int ret = avcodec_receive_frame(m_codecContext, m_frame);
    if(ret < 0)
    {
    av_frame_unref(m_frame);
    if(readRet < 0)
    {
    m_end = true; // 当无法读取到AVPacket并且解码器中也没有数据时表示读取完成
    }
    return QImage();
    } m_pts = m_frame->pts; // 为什么图像转换上下文要放在这里初始化呢,是因为m_frame->format,如果使用硬件解码,解码出来的图像格式和m_codecContext->pix_fmt的图像格式不一样,就会导致无法转换为QImage
    if(!m_swsContext)
    {
    // 获取缓存的图像转换上下文。首先校验参数是否一致,如果校验不通过就释放资源;然后判断上下文是否存在,如果存在直接复用,如不存在进行分配、初始化操作
    m_swsContext = sws_getCachedContext(m_swsContext,
    m_frame->width, // 输入图像的宽度
    m_frame->height, // 输入图像的高度
    (AVPixelFormat)m_frame->format, // 输入图像的像素格式
    m_size.width(), // 输出图像的宽度
    m_size.height(), // 输出图像的高度
    AV_PIX_FMT_RGBA, // 输出图像的像素格式
    SWS_BILINEAR, // 选择缩放算法(只有当输入输出图像大小不同时有效),一般选择SWS_FAST_BILINEAR
    nullptr, // 输入图像的滤波器信息, 若不需要传NULL
    nullptr, // 输出图像的滤波器信息, 若不需要传NULL
    nullptr); // 特定缩放算法需要的参数(?),默认为NULL
    if(!m_swsContext)
    {
    #if PRINT_LOG
    qWarning() << "sws_getCachedContext() Error!";
    #endif
    free();
    return QImage();
    }
    } // AVFrame转QImage
    uchar* data[] = {m_buffer};
    int lines[4];
    av_image_fill_linesizes(lines, AV_PIX_FMT_RGBA, m_frame->width); // 使用像素格式pix_fmt和宽度填充图像的平面线条大小。
    ret = sws_scale(m_swsContext, // 缩放上下文
    m_frame->data, // 原图像数组
    m_frame->linesize, // 包含源图像每个平面步幅的数组
    0, // 开始位置
    m_frame->height, // 行数
    data, // 目标图像数组
    lines); // 包含目标图像每个平面的步幅的数组
    QImage image(m_buffer, m_frame->width, m_frame->height, QImage::Format_RGBA8888);
    av_frame_unref(m_frame); return image;
    } /**
    * @brief 关闭视频播放并释放内存
    */
    void VideoDecode::close()
    {
    clear();
    free(); m_totalTime = 0;
    m_videoIndex = 0;
    m_totalFrames = 0;
    m_obtainFrames = 0;
    m_pts = 0;
    m_frameRate = 0;
    m_size = QSize(0, 0);
    } /**
    * @brief 视频是否读取完成
    * @return
    */
    bool VideoDecode::isEnd()
    {
    return m_end;
    } /**
    * @brief 返回当前帧图像播放时间
    * @return
    */
    const qint64 &VideoDecode::pts()
    {
    return m_pts;
    } /**
    * @brief 显示ffmpeg函数调用异常信息
    * @param err
    */
    void VideoDecode::showError(int err)
    {
    #if PRINT_LOG
    memset(m_error, 0, ERROR_LEN); // 将数组置零
    av_strerror(err, m_error, ERROR_LEN);
    qWarning() << "DecodeVideo Error:" << m_error;
    #else
    Q_UNUSED(err)
    #endif
    } /**
    * @brief 将AVRational转换为double,用于计算帧率
    * @param rational
    * @return
    */
    qreal VideoDecode::rationalToDouble(AVRational* rational)
    {
    qreal frameRate = (rational->den == 0) ? 0 : (qreal(rational->num) / rational->den);
    return frameRate;
    } /**
    * @brief 清空读取缓冲
    */
    void VideoDecode::clear()
    {
    // 因为avformat_flush不会刷新AVIOContext (s->pb)。如果有必要,在调用此函数之前调用avio_flush(s->pb)。
    if(m_formatContext && m_formatContext->pb)
    {
    avio_flush(m_formatContext->pb);
    }
    if(m_formatContext)
    {
    avformat_flush(m_formatContext); // 清理读取缓冲
    }
    } void VideoDecode::free()
    {
    // 释放上下文swsContext。
    if(m_swsContext)
    {
    sws_freeContext(m_swsContext);
    m_swsContext = nullptr; // sws_freeContext不会把上下文置NULL
    }
    // 释放编解码器上下文和与之相关的所有内容,并将NULL写入提供的指针
    if(m_codecContext)
    {
    avcodec_free_context(&m_codecContext);
    }
    // 关闭并失败m_formatContext,并将指针置为null
    if(m_formatContext)
    {
    avformat_close_input(&m_formatContext);
    }
    if(m_packet)
    {
    av_packet_free(&m_packet);
    }
    if(m_frame)
    {
    av_frame_free(&m_frame);
    }
    if(m_buffer)
    {
    delete [] m_buffer;
    m_buffer = nullptr;
    }
    }

6、完整源代码

Qt-FFmpeg开发-视频播放【软解码】(1)的更多相关文章

  1. QT+FFMPEG实现视频播放

    开发环境:MinGW+QT5.9+FFMPEG20190212 一.开发环境搭建 FFMPEG的开发环境部署比如容易,在官网下载库文件,然后在QT里面指定路径,把相关dll文件放到exe目录下就可以了 ...

  2. Qt+FFmpeg 简单实现视频播放

    这里使用 Qt + FFmpeg 实现了一个简单播放视频的例子.先看下按下按钮播放视频时的效果图: 完整工程下载链接:Github-FFmpeg_demo 注意:一定要将 bin 目录下的 dll 文 ...

  3. 海思3519 qt ffmpeg 软解码播放avi

    在海思3519上基于qt采用ffmpeg对avi进行解码显示,其中ffmpeg的配置,qt的配置在前文中已经说明,在此不再赘述. 解码 解码在单独的线程中进行,具体的代码如下: void VideoP ...

  4. FFmpeg开发笔记(四):ffmpeg解码的基本流程详解

    若该文为原创文章,未经允许不得转载原博主博客地址:https://blog.csdn.net/qq21497936原博主博客导航:https://blog.csdn.net/qq21497936/ar ...

  5. FFmpeg开发笔记(五):ffmpeg解码的基本流程详解(ffmpeg3新解码api)

    若该文为原创文章,未经允许不得转载原博主博客地址:https://blog.csdn.net/qq21497936原博主博客导航:https://blog.csdn.net/qq21497936/ar ...

  6. FFmpeg开发笔记(九):ffmpeg解码rtsp流并使用SDL同步播放

    前言   ffmpeg播放rtsp网络流和摄像头流.   Demo   使用ffmpeg播放局域网rtsp1080p海康摄像头:延迟0.2s,存在马赛克     使用ffmpeg播放网络rtsp文件流 ...

  7. FFmpeg开发笔记(十):ffmpeg在ubuntu上的交叉编译移植到海思HI35xx平台

    FFmpeg和SDL开发专栏(点击传送门) 上一篇:<FFmpeg开发笔记(九):ffmpeg解码rtsp流并使用SDL同步播放>下一篇:敬请期待   前言   将ffmpeg移植到海思H ...

  8. [转]FFMPEG视音频编解码零基础学习方法

    在CSDN上的这一段日子,接触到了很多同行业的人,尤其是使用FFMPEG进行视音频编解码的人,有的已经是有多年经验的“大神”,有的是刚开始学习的初学者.在和大家探讨的过程中,我忽然发现了一个问题:在“ ...

  9. [总结]FFMPEG视音频编解码零基础学习方法--转

    ffmpeg编解码学习   目录(?)[-] ffmpeg程序的使用ffmpegexeffplayexeffprobeexe 1 ffmpegexe 2 ffplayexe 3 ffprobeexe ...

  10. FFMPEG视音频编解码零基础学习方法

    在CSDN上的这一段日子,接触到了很多同行业的人,尤其是使用FFMPEG进行视音频编解码的人,有的已经是有多年经验的“大神”,有的是刚开始学习的初学者.在和大家探讨的过程中,我忽然发现了一个问题:在“ ...

随机推荐

  1. (四) 一文搞懂 JMM - 内存模型

    4.JMM - 内存模型 1.JMM内存模型 JMM与happen-before 1.可见性问题产生原因 下图为x86架构下CPU缓存的布局,即在一个CPU 4核下,L1.L2.L3三级缓存与主内存的 ...

  2. 多线程/GIL全局锁

    目录 线程理论 创建线程的两种方式 线程的诸多特性 GIL全局解释器 验证GIL存在 同一个进程下多线程是否有优势 死锁现象 信号量 Event事件 线程理论 进程 进程其实是资源单位 标示开辟一块内 ...

  3. C#关于委托的一些事,开发日志

    ----- 委托是什么------ 其实委托事件很好理解,就当成是c语言中的函数指针或者是回调函数,或者说换种理解方式,信号和槽?触发器和接收器?总之就是一个地方调用了这个函数,那么在另一个地方也会调 ...

  4. 事件 jQuery类库、Bootstrap页面框架

    目录 jQuery查找标签 基本选择器 组合选择器 层级选择器 属性选择器 基本筛选器 表单筛选器 筛选器方法 链式的本质(jQuery一行代码走天下) 操作标签 class操作 位置操作 文本操作 ...

  5. 关于Git在Visual studio及Git Bush中的日常操作教程,有图有说明,会一直更新本页内容... (Git相对SVN具有更加安全的分布式存储, 分支版本之间切换秒级速度, 分支版本强大灵活等特点)

    >安装命令行和TortoiseGit UI程序. <git bash的安装> https://git-scm.com/downloads <windows可视化工具 Torto ...

  6. cmd/批处理常用命令

    启动新窗口执行命令 ::执行完毕以后,新开的窗口不会自动关闭 start cmd /k echo 123 ::执行完毕以后,新开的窗口会自动关闭 start cmd /C "echo 123 ...

  7. 【大型软件开发】浅谈大型Qt软件开发(一)开发前的准备——在着手开发之前,我们要做些什么?

    前言 最近我们项目部的核心产品正在进行重构,然后又是年底了,除了开发工作之外项目并不紧急,加上加班时间混不够了....所以就忙里偷闲把整个项目的开发思路聊一下,以供参考. 鉴于接下来的一年我要操刀这个 ...

  8. 《Effective C++》模版与泛型编程

    Item41:了解隐式接口和编译期多态. 纵使你从未使用过templates,应该不陌生"运行期多态"和"编译期多态"之间的差异.因为它类似于"哪一个 ...

  9. 靶机练习 - 温故知新 - Toppo(sudo 提权)

    重新做了一下以前做过的第一个靶机(https://www.cnblogs.com/sallyzhang/p/12792042.html),这个靶机主要是练习sudo提权,当时不会也没理解. 开启靶机, ...

  10. angular引入http服务创建服务注入