Qt线程封装FFmpeg播放器类
介绍
工作开发中需要处理的文件很多并无音频,针对这一场景,这里分享工作中自己封装使用的类库。精简的代码实现了播放、暂停、停止、快进、快退、进度跳转、倍速播放功能。直接放代码,方便后期复制使用。
代码
头文件
/*** @file videoplayer.h* @brief ffmpeg实现视频解码* @author ZXT* @date 2023.12.23*/#ifndef VIDEOPLAYER_H
#define VIDEOPLAYER_H#include <QThread>
#include <QDebug>
#include <QImage>extern "C"
{
#include "libavformat/avformat.h"
#include "libavcodec/avcodec.h"
#include "libswscale/swscale.h"
#include "libavutil/imgutils.h"
#include "libavutil/time.h"
}class VideoDecoder : public QThread
{Q_OBJECT
public:explicit VideoDecoder(QObject *parent = nullptr);~VideoDecoder();/*** @brief 开始播放* @param path入参 路径*/void startPlay(const QString &path);/*** @brief 停止播放*/void stopPlay();/*** @brief 暂停或继续播放* @param pause入参 ture暂停 false继续*/void pausePlay(bool pause);/*** @brief 进度跳转播放* @param sec入参 跳转的秒数*/void seekPlay(int sec);/*** @brief 倍速播放* @param speed入参 播放速度*/void speedPlay(float speed = 1.0);signals:/*** @brief 播放时长* @param sec入参 时长秒数*/void sigDuration(int sec);/*** @brief 播放位置* @param sec入参 当前秒数*/void sigPlayPosition(int sec);/*** @brief 播放结束* @param ret出参 状态码*/void sigPlayFinished(int ret);/*** @brief 发送解码后的显示图像* @param image出参 视频图像*/void sigSendImage(const QImage &image);protected:void run();private://运行标志volatile bool m_isRun = false;//暂停状态volatile bool m_pause = false;//进度跳转状态volatile bool m_seek = false;//优化跳转速度volatile bool m_seekFilter = false;//文件路径QString m_filePath;//开始时间 ms单位int64_t m_startTime = 0;//暂停时间 ms单位int64_t m_pauseTime = 0;//跳转时间 ms单位int64_t m_seekTime = 0;//时长信息 秒int m_videoDuration = 0;//播放速率float m_speedValue = 1.0;//上一次的播放速率float m_lastSpeedValue = 1.0;
};#endif // VIDEOPLAYER_H
实现文件
#include "videoplayer.h"VideoDecoder::VideoDecoder(QObject *parent) : QThread(parent)
{}VideoDecoder::~VideoDecoder()
{quit();wait();
}//开始播放
void VideoDecoder::startPlay(const QString &path)
{m_filePath = path;m_isRun = true;m_pause = false;m_seek = false;m_speedValue = 1.0;m_lastSpeedValue = 1.0;this->start();
}//停止播放
void VideoDecoder::stopPlay()
{m_pause = false;m_seek = false;m_isRun = false;
}//暂停或继续播放
void VideoDecoder::pausePlay(bool pause)
{m_pause = pause;if(pause){m_pauseTime = av_gettime_relative() / 1000;}else{int offset = av_gettime_relative() / 1000 - m_pauseTime;m_startTime += offset;}
}//进度跳转播放
void VideoDecoder::seekPlay(int sec)
{if(!m_isRun)return;if(m_videoDuration == sec)sec -= 2;m_seekTime = sec * 1000;m_seekFilter = true;m_seek = true;
}//倍速播放
void VideoDecoder::speedPlay(float speed)
{int64_t elapsed = av_gettime_relative() / 1000 - m_startTime;int offset = elapsed - (elapsed * this->m_lastSpeedValue / speed);m_startTime += offset;m_speedValue = speed;m_lastSpeedValue = speed;
}void VideoDecoder::run()
{qDebug() << "VideoDecoder start" << m_filePath;std::string temp = m_filePath.toStdString();AVFormatContext *inFmtCtx = avformat_alloc_context();int ret = avformat_open_input(&inFmtCtx, temp.c_str(), NULL, NULL);if (ret < 0){qDebug() << "open input error";emit sigPlayFinished(-1);return;}//获取流信息ret = avformat_find_stream_info(inFmtCtx, NULL);if (ret < 0){qDebug() << "find stream info error";emit sigPlayFinished(-1);return;}//获取视频流信息 目前只有视频流bool getVideo = false;int videoIndex = av_find_best_stream(inFmtCtx, AVMEDIA_TYPE_VIDEO, -1, -1, NULL, 0);AVStream *videoStream = NULL;AVCodec *videoDecoder = NULL;AVCodecContext *videoDeCodecCtx = NULL;if (videoIndex >= 0){videoStream = inFmtCtx->streams[videoIndex];//初始化解码器videoDecoder = avcodec_find_decoder(videoStream->codecpar->codec_id);videoDeCodecCtx = avcodec_alloc_context3(videoDecoder);if(videoDeCodecCtx != NULL){avcodec_parameters_to_context(videoDeCodecCtx, videoStream->codecpar);ret = avcodec_open2(videoDeCodecCtx, videoDecoder, NULL);if(ret < 0)avcodec_free_context(&videoDeCodecCtx);elsegetVideo = true;}}if(!getVideo){avformat_close_input(&inFmtCtx);emit sigPlayFinished(-1);return;}AVFrame *swsFrame = av_frame_alloc();SwsContext *swsCtx = nullptr;uint8_t *videoData = nullptr;//输出视频参数信息if(getVideo){int srcW = videoStream->codecpar->width;int srcH = videoStream->codecpar->height;AVPixelFormat format = videoDeCodecCtx->pix_fmt;m_videoDuration = inFmtCtx->duration / AV_TIME_BASE;emit sigDuration(m_videoDuration);int byte = av_image_get_buffer_size(AV_PIX_FMT_RGB32, srcW, srcH, 1);videoData = (uint8_t *)av_malloc(byte * sizeof(uint8_t));av_image_fill_arrays(swsFrame->data, swsFrame->linesize, videoData, (AVPixelFormat)AV_PIX_FMT_RGB32, srcW, srcH, 1);swsCtx = sws_getContext(srcW, srcH, (AVPixelFormat)format, srcW, srcH, (AVPixelFormat)AV_PIX_FMT_RGB32, SWS_FAST_BILINEAR, NULL, NULL, NULL);}//开始时刻m_startTime = av_gettime_relative() / 1000;int64_t ptsTime = 0;int64_t ptsBackup = 0;int curPlayPos = 0;AVPacket *packet = av_packet_alloc();AVFrame *videoFrame = av_frame_alloc();while(m_isRun){//暂停if(m_pause){QThread::msleep(200);continue;}//进度切换if(m_seek){//跳转的播放时刻 单位微秒int64_t timeStamp = m_seekTime * 1000;if (inFmtCtx->start_time != AV_NOPTS_VALUE){timeStamp += inFmtCtx->start_time;}//注:seek若关键帧间隔大需避免延时timeStamp = av_rescale_q(timeStamp, AVRational{1, AV_TIME_BASE}, videoStream->time_base);ret = av_seek_frame(inFmtCtx, videoIndex, timeStamp, AVSEEK_FLAG_BACKWARD);if(ret < 0){qDebug() << "av_seek_frame fail" << m_seekTime;}else{//清空内部帧队列if(videoDeCodecCtx)avcodec_flush_buffers(videoDeCodecCtx);//调整时钟int64_t offset = m_seekTime - ptsTime;m_startTime -= offset;}m_seek = false;}//不断读取packetret = av_read_frame(inFmtCtx, packet);if (ret == AVERROR_EOF){m_isRun = false;break;}if(packet->stream_index == videoIndex){//编码数据进行解码ret = avcodec_send_packet(videoDeCodecCtx, packet);if (ret < 0){av_packet_unref(packet);continue;}ret = avcodec_receive_frame(videoDeCodecCtx, videoFrame);if (ret < 0){av_packet_unref(packet);continue;}//计算当前帧实际时间 msptsTime = videoFrame->pts * av_q2d(videoStream->time_base) * 1000;if(m_seekFilter){//跳转播放时间不符合的帧直接丢弃 默认阈值200msint offset = m_seekTime - ptsTime;if(0 > offset || offset < 200){m_seekFilter = false;}else{av_frame_unref(videoFrame);av_packet_unref(packet);continue;}}//倍速将改变原pts值 太高倍速会导致解码消耗过高、渲染过频繁可考虑抽帧ptsBackup = ptsTime;ptsTime *= (1 / m_speedValue);//控制速度 ms单位qint64 elapsed = av_gettime_relative() / 1000 - m_startTime;int64_t sleepMs = ptsTime - elapsed;if(sleepMs > 3){QThread::msleep(sleepMs);}//发送播放位置信息int sec = ptsBackup / 1000;if(sec != curPlayPos){curPlayPos = sec;emit sigPlayPosition(curPlayPos);}//将解码后的frame数据转换为Imagesws_scale(swsCtx, (const uint8_t *const *)videoFrame->data, videoFrame->linesize, 0, videoFrame->height, swsFrame->data, swsFrame->linesize);QImage image((uchar *)videoData, videoFrame->width, videoFrame->height, QImage::Format_RGB32);QImage copy = image.copy();emit sigSendImage(copy);av_frame_unref(videoFrame);}av_packet_unref(packet);}//释放资源sws_freeContext(swsCtx);av_frame_free(&swsFrame);av_free(videoData);av_packet_free(&packet);av_frame_free(&videoFrame);avcodec_free_context(&videoDeCodecCtx);avformat_close_input(&inFmtCtx);emit sigPlayFinished(0);qDebug() << "VideoDecoder end" << m_filePath;return;
}
相关文章:
Qt线程封装FFmpeg播放器类
介绍 工作开发中需要处理的文件很多并无音频,针对这一场景,这里分享工作中自己封装使用的类库。精简的代码实现了播放、暂停、停止、快进、快退、进度跳转、倍速播放功能。直接放代码,方便后期复制使用。 代码 头文件 /*** file videopla…...
git 常用命令 修改 远程仓库 默认分支
git remote set-head origin -a 将 origin/HEAD 指向 远程仓库的 默认分支(-a 即 --auto) git remote set-head origin dev 将 origin/HEAD 指向 (origin/dev) git remote set-head origin -d 删除 origin/…...
springboot项目禁用dataSource数据源功能,只需修改yml文件,关闭数据库连接功能
当我们遇到以下报错: Description: Failed to configure a DataSource: url attribute is not specified and no embedded datasource could be configured. Reason: Failed to determine a suitable driver class Action: Consider the following: If you …...
【Vue.js设计与实现解读-1】
Vue设计与实现阅读-1 1、命令式和声明式2、性能3、虚拟DOM性能4、运行时和编译时5、总结 前言 最近工作清闲了些,想着很久没有看书,Vue.js设计与实现这本书看了好几次都没有读完,趁着这个机会边读边记录一下吧。如果有理解的不正确的地方&…...
苗情生态自动监测系统-科普知识
随着科技的飞速发展,智能化技术在各个领域的应用越来越广泛。在农业领域,苗情生态自动监测系统的出现,为农业生产带来了革命性的变革。它不仅能够实时监测植物的生长状况,还能对环境因素进行全面监控,为农业生产提供科…...
test 系统学习-04-test converate 测试覆盖率 jacoco 原理介绍
测试覆盖率 测试覆盖率(test coverage)是衡量软件测试完整性的一个重要指标。掌握测试覆盖率数据,有利于客观认识软件质量,正确了解测试状态,有效改进测试工作。 当然,要发挥这些作用,前提是我们掌握了真实的测试覆盖…...
小型企业成为网络犯罪分子获取数据的目标
在过去十年的大部分时间里,网络犯罪的巨额资金来自针对大型组织的勒索软件攻击。这种威胁仍然存在。但犯罪分子可能会将注意力转向中小企业 (SMB)。这对消费者的影响将是巨大的。 将软件即服务 (SaaS) 技术用于核心业务功能继续将中小企业整合到全球供应链中。由于…...
PyTorch的Tensor(张量)
一、Tensor概念 什么是张量? 张量是一个多维数组,它是标量、向量、矩阵的高维拓展 Tensor与Variable Variable是torch.autograd中的数据类型,主要用于封装Tensor,进行自动求导。 data: 被包装的Tensorgrad: data的梯度&…...
spug发布问题汇总记录
问题导览 1. [vite]: Rollup failed to resolve import "element-plus" from "src/main.js". 项目框架简介 vue3viteelement-plus 解决方案 - 1. 配置淘宝镜像源:npm config set registry https://registry.npm.taobao.org/ - 2. npm inst…...
SpringBoot-搭建集成Mybatis的项目
本文介绍了如何在IntelliJ IDEA中使用SpringBoot和Mybatis构建Java Web应用程序。通过本文的学习,读者将了解如何使用IntelliJ IDEA快速搭建一个基于SpringBoot和Mybatis的Java Web应用程序,提高开发效率。IntelliJ IDEA是一款功能强大的Java集成开发环境…...
mysql隐式转换规则
MySQL 中的隐式类型转换发生在比较操作或者其他一些需要特定数据类型参数的上下文中,如果参与操作的表达式或列的数据类型不匹配,MySQL 就会自动进行数据类型转换以适配预期的数据类型。 以下是 MySQL 的一些常见隐式转换规则: 字符串和数字…...
怎么解决 Nginx反向代理加载速度慢?
Nginx反向代理加载速度慢可能由多种原因引起,以下是一些可能的解决方法: 1,网络延迟: 检查目标服务器的网络状况,确保其网络连接正常。如果目标服务器位于不同的地理位置,可能会有较大的网络延迟。考虑使用…...
Eureka工作原理超详细讲解介绍
Eureka 是 Netflix 开源的一款服务注册与发现框架,主要用于构建分布式系统中的服务治理和负载均衡。下面是关于 Eureka 工作原理的详细介绍:1.Eureka 架构: Eureka 采用了客户端-服务器架构,包括 Eureka Server 和 Eureka Client …...
SQL WHERE 语句(条件选择)
WHERE 子句用于过滤记录。 SQL WHERE 子句 WHERE 子句用于提取那些满足指定条件的记录。 SQL WHERE 语法 SELECT column1, column2, ... FROM table_name WHERE condition; 参数说明: column1, column2, ...:要选择的字段名称,可以为多…...
用UCLI(TCL)控制verdi dump 波形
UCLI(Unified Command-line Interface)为Synopsys验证工具了提供一组通用命令,通过UCLI可以执行任意TCL(Tool Command Language)命令。在我们的验证环境中,通常跟ucli打交道的地方是用来控制开始dump和结束…...
如何使用 Python+selenium 进行 web 自动化测试?
Selenium是一个自动化测试工具,它可以模拟用户在浏览器中的操作,比如点击、输入、选择等等。它支持多种浏览器,包括Chrome、Firefox、Safari等等,并且可以在多个平台上运行。 安装和配置Selenium 在使用Selenium之前,…...
约瑟夫问题
约瑟夫问题 题目描述 n n n 个人围成一圈,从第一个人开始报数,数到 m m m 的人出列,再由下一个人重新从 1 1 1 开始报数,数到 m m m 的人再出圈,依次类推,直到所有的人都出圈,请输出依次出圈人的编号。…...
文件管理方法:利用文件大小进行筛选,高效移动文件至目标文件夹
在日常工作中,文件管理是一项至关重要的任务。为了更高效地管理文件,可以利用文件大小进行筛选,并将文件快速移动至目标文件夹。接下来一起来看看云炫文件管理器如何利用文件大小进行筛选,以及如何高效移动文件至目标文件夹的方法…...
python报错:TypeError: Descriptors cannot be created directly.
问题 报错提示: TypeError:不能直接创建描述符。 如果此调用来自 _pb2.py 文件,则您生成的代码已过期,必须使用 protoc > 3.19.0 重新生成。 如果您不能立即重新生成原型,其他一些可能的解决方法是: 1.…...
Linux 内核调试
文章目录 一、方法论 一、方法论 qemu 虚拟机 Linux内核学习 Linux 内核调试 一:概述 Linux 内核调试 二:ubuntu20.04安装qemu Linux 内核调试 三:《QEMU ARM guest support》翻译 Linux 内核调试 四:qemu-system-arm功能选项整…...
Prometheus-AlertManager 邮件告警
环境,软件准备 本次演示环境,我是在虚拟机上安装 Linux 系统来执行操作,以下是安装的软件及版本: System: CentOS Linux release 7.6Docker: 24.0.5Prometheus: v2.37.6Consul: 1.6.1 docker 安装prometheus,alertmanage,说明一下这里直接将…...
Volcano Controller控制器源码解析
Volcano Controller控制器源码解析 本文从源码的角度分析Volcano Controller相关功能的实现。 本篇Volcano版本为v1.8.0。 Volcano项目地址: https://github.com/volcano-sh/volcano controller命令main入口: cmd/controller-manager/main.go controller相关代码目录: pkg/co…...
开源协议简介和选择
软件国产化已经提到日程上了,先来研究一下开源协议。 引言 在追求“自由”的开源软件领域的同时不能忽视程序员的权益。为了激发程序员的创造力,现今世界上有超过60种的开源许可协议被开源促进组织(Open Source Initiative)所认可…...
大创项目推荐 深度学习卫星遥感图像检测与识别 -opencv python 目标检测
文章目录 0 前言1 课题背景2 实现效果3 Yolov5算法4 数据处理和训练5 最后 0 前言 🔥 优质竞赛项目系列,今天要分享的是 🚩 **深度学习卫星遥感图像检测与识别 ** 该项目较为新颖,适合作为竞赛课题方向,学长非常推荐…...
pod的环节
pod 是k8s当中最小的资源管理组件 Pod也是最小化运行容器化的应用的资源管理对象 Pod是一个抽象化的概念,可以理解为一个或多个容器化的集合 在一个pod当中运行一个容器,是最常用的方式 在一个pod当中同时运行多个容器,在一个pod当中可以…...
Unity | Shader基础知识番外(向量数学知识速成)
目录 一、向量定义 二、计算向量 三、向量的加法(连续行走) 四、向量的长度 五、单位向量 六、向量的点积 1 计算 2 作用 七、向量的叉乘 1 承上启下 2 叉乘结论 3 叉乘的计算(这里看不懂就百度叉乘计算) 八、欢迎收…...
一个小白的微不足道的见解关于未来
随着科技的不断发展,IT行业日益壮大,运维工程师在其中扮演着至关重要的角色。他们负责维护和管理企业的技术基础设施,确保系统的正常运行。然而,随着技术的进步和行业的变化,运维工程师的未来将面临着一系列挑战和机遇…...
图的遍历(搜索)算法(深度优先算法DFS和广度优先算法BFS)
一、图的遍历的定义: 从图的某个顶点出发访问遍图中所有顶点,且每个顶点仅被访问一次。(连通图与非连通图) 二、深度优先遍历(DFS); 1、访问指定的起始顶点; 2、若当前访问的顶点…...
抖店做不起来?新手常见起店失败问题总结,看下你中了几条?
我是王路飞。 能看到这篇文章的,肯定是处境符合标题内容了。 抖店的门槛很低,运营思路其实也不算难,但就是很多新手做不起来。 这中间,可能跟平台、项目没什么关系,而是跟你自己有关系,走错了方向&#…...
【每日面试题】精选java面试题之redis
Redis是什么?为什么要使用Redis? Redis是一个开源的高性能键值对存储数据库。它提供了多种数据结构,包括字符串、列表、集合、有序集合、哈希表等。Redis具有快速、可扩展、持久化、支持多种数据结构等特点,适用于缓存、消息队列…...
微网站 功能/上海百度推广方案
八周二次课(1月30日)10.28 rsync工具介绍rsync -av /etc/passwd /tmp/1.txt 把passwdcp到tmp下并命名为1.txtrsync -av /etc/passwd root192.168.133.130:/tmp/1.txt cp到130的root的tmp目录SRC 源目录 DEST 目标目录 [user]host: DEST user可以省略&a…...
58同城宿迁二手房/太原百度网站快速优化
Redis五大基本Value数据结构类型: String,List,Hash,Set,ZSet1:String String 是Redis 里面最简单的一种数据结构。在Redis中,所有的Key都是字符串,存储对象时也是需要进行序列化操作,但是不同…...
厦门外贸网站建设多少钱/百度如何做推广
arpspoof是一个好用的ARP欺骗工具,Kali linux中自带了该工具,在ubuntu中,安装它只需运行命令:sudo apt-get install dsniff安装完成后,输入命令:man arpspoof 可以查看使用手册,2.4版本的手册内…...
刚建设的网站多久能在百度查到/如何刷关键词指数
#sudo vim /etc/ssh/sshd_config 找到并用#注释掉这行:PermitRootLogin prohibit-password 新建一行 添加:PermitRootLogin yes 重启服务 #sudo service ssh restart sudo passwd root #设置密码 然后ssh root192.168.2.21就可以登录了...
做网站老板嫌弃太丑谁的锅/网页设计用什么软件做
1.内存溢出相对于内存泄漏来说,尽管更容易被理解,但是同样的,内存溢出也是引发程序崩溃的罪魁祸首之一。 2.由于GC一致在发展,所有一般情况先,除非应用程序占用的内存增长速度非常快,造成垃圾回收根不上内存…...
如何判断网站做的关键词/宁德市区哪里好玩
此功能是通过shell32.dll中一个索引号为60的API函数调用,显示"关闭Windows"对话窗口实现的。具体方法为IDC_SHUTDOWNCOMPUTER按钮添加BN_CLICKED消息处理函数:void CControlDlg::OnShutdowncomputer() { HINSTANCE hInstLoadLibrary("…...