Qt边推流边录制/实时性好延迟低/16路1080P推流加录制只占1%CPU
nanshan 2024-11-23 20:15 134 浏览 0 评论
一、前言
这个一边推流一边录制的功能,有很多用户提到过,之前因为时间的原因,一直没有搞,年初的时候索性抽空搞了下,也着实费了些功夫。推流用的是ffmpeg这个开源的牛逼的第三方库,搞音视频开发的人应该没人不认识这个库,养活了很多程序员以及厂家,甚至不乏一些大厂,如果能把ffmpeg搞精通,在国内拿个30K以上毫无压力分分钟的事情。但是这个库也确实有一定的难度,起码对我来说还是挺难的,按照提供的示例做点简单的demo没有问题,要搞稳定搞兼容性以及兼容各种需求场景,就非常难了,没有个好几年的捣鼓,很难搞好。
推流的前提是拉流,之前已经用ffmpeg做了拉流和保存,既可以保存到本地视频文件,也可以保存到rtsp/rtmp这种地址,保存到流地址其实就是推流,以前没搞过的时候还以为多复杂,原来就是保存文件改个地址,总共就改动几行代码就行。既然已经可以推流和保存,那说明一边推流一边录制也是可行的。为了追求最简方式实现,通过在原来的基础上,增加一个信号,也就是在保存的时候发出去一个avpacket的包信号,这个包是最终要写入到文件的数据包,如果源头就是264/265这种格式的数据,这个包可以直接存入文件即可。于是一边推一边存的功能,就是重新new一个ffmpegsave类,将之前推流的类发出来的收到数据包信号,发给存储类的槽即可,非常简单方便易用。打完收工完美实现,又是去沙县加鸡腿的一天。
二、效果图
三、体验地址
- 国内站点:https://gitee.com/feiyangqingyun
- 国际站点:https://github.com/feiyangqingyun
- 个人作品:https://blog.csdn.net/feiyangqingyun/article/details/97565652
- 体验地址:https://pan.baidu.com/s/1d7TH_GEYl5nOecuNlWJJ7g 提取码:01jf 文件名:bin_video_push。
- 视频主页:https://space.bilibili.com/687803542
四、功能特点
- 支持各种本地音视频文件和网络音视频文件,格式包括mp3、aac、wav、wma、mp4、mkv、rmvb、wmv、mpg、flv、asf等。
- 支持各种网络音视频流,网络摄像头,协议包括rtsp、rtmp、http等。
- 支持本地摄像头设备推流,可指定分辨率、帧率、格式等。
- 支持本地桌面采集推流,可指定屏幕索引、采集区域、起始坐标、帧率等,也支持指定窗口标题进行采集。
- 可实时切换预览视频文件,可切换音视频文件播放进度,切换到哪里就推流到哪里。预览过程中可以切换静音状态和暂停推流。
- 可指定重新编码推流,任意源头格式可选强转264或265格式。
- 可转换分辨率推流,设置等比例缩放或者指定分辨率进行转换。
- 推流的清晰度、质量、码率都可调,可以节约网络带宽和拉流端的压力。
- 音视频文件自动循环不间断推流。
- 音视频流有自动掉线重连机制,重连成功自动继续推流。
- 支持各种流媒体服务程序,包括但不限于mediamtx、ZLMediaKit、srs、LiveQing、nginx-rtmp、EasyDarwin、ABLMediaServer。
- 通过配置文件自动加载对应流媒体程序的协议和端口,自动生成推流地址和各种协议的拉流地址。可以通过配置文件自己增加流媒体程序。
- 可选rtmp、rtmp格式推流,推流成功后,支持多种格式拉流,包括但不限于rtsp、rtmp、hls、flv、ws-flv、webrtc等。
- 在软件上推流成功后,可以直接单击网页预览,实时预览推流后拉流的画面,多画面网页展示。
- 软件界面上可单击对应按钮,动态添加文件和目录,可手动输入地址。
- 推拉流实时性极高,延迟极低,延迟时间大概在100ms左右。
- 极低CPU资源占用,4路主码流推流只需要占用0.2%CPU。理论上常规普通PC机器推100路毫无压力,主要性能瓶颈在网络。
- 可以推流到外网服务器,然后通过手机、电脑、平板等设备播放对应的视频流。
- 每路推流都可以手动指定唯一标识符(方便拉流/用户无需记忆复杂的地址),没有指定则按照策略随机生成hash值。也支持自动按照指定标识后面加数字的方式递增命名。比如设置标识为字母v,策略为标识递增,则每添加一个对应的推流码命名依次是v1、v2、v3等。
- 根据推流协议自动转码格式,默认策略按照选择的推流协议,比如rtsp支持265而rtmp不支持,如果是265的文件而选择rtmp推流,则自动转码成264格式再推流。
- 音视频同步推流,在拉流和采集的时候就会自动处理好同步,同步后的数据再推流。
- 表格中实时显示每一路推流的分辨率和音视频数据状态,灰色表示没有输入流,黑色表示没有输出流,绿色表示原数据推流,红色表示转码后的数据推流。
- 自动重连视频源,自动重连流媒体服务器,保证启动后,推流地址和打开地址都实时重连,只要恢复后立即连上继续采集和推流。
- 根据不同的流媒体服务器类型,自动生成对应的rtsp、rtmp、hls、flv、ws-flv、webrtc拉流地址,用户可以直接复制该地址到播放器或者网页中预览查看。
- 添加的推流地址等信息自动存储到文件,可以手动打开进行修改,默认启动后自动加载历史记录。
- 可以指定生成的网页文件保存位置,方便作为网站网页发布,可以直接在浏览器中输入网址进行访问,发布后可以直接在局域网其他设备比如手机或者电脑打开对应网址访问。
- 可选是否开机启动、后台运行等。网络推流添加的rtsp地址可勾选是否隐藏地址中的用户信息。
- 自带设备推流模块,自动识别本地设备,包括本地的摄像头和桌面,可以手动选择不同的是视频和音频采集设备进行推流。
- 自带文件点播模块,添加文件后用户可以拉取地址点播,用户端可以任意切换播放进度。支持各种浏览器(谷歌chromium、微软edge、火狐firefox等)、各种播放器(vlc、mpv、ffplay、potplayer、mpchc等)打开请求。
- 文件点播模块实时统计显示每个文件对应的访问数量、总访问数量、不同IP地址访问数量。
- 文件点播模块采用纯QTcpSocket通信,不依赖流媒体服务程序,核心源码不到500行,注释详细,功能完整。
- 支持任意Qt版本(Qt4、Qt5、Qt6),支持任意系统(windows、linux、macos、android、嵌入式linux等)。
五、相关代码
#include "netpushclient.h"
#include "ffmpegthread.h"
#include "ffmpegsave.h"
#include "videohelper.h"
#include "osdgraph.h"
bool NetPushClient::checkB = false;
bool NetPushClient::recordInteger = false;
int NetPushClient::recordDuration = 0;
int NetPushClient::encodeVideo = 0;
float NetPushClient::encodeVideoRatio = 1;
QString NetPushClient::encodeVideoScale = "1";
NetPushClient::NetPushClient(QObject *parent) : QObject(parent)
{
ffmpegThread = NULL;
ffmpegSave = NULL;
//定时器控制多久录制一个文件
timerRecord = new QTimer(this);
timerRecord->setInterval(1000);
connect(timerRecord, SIGNAL(timeout()), this, SLOT(checkRecord()));
}
NetPushClient::~NetPushClient()
{
this->stop();
}
QString NetPushClient::getMediaUrl()
{
return this->mediaUrl;
}
QString NetPushClient::getPushUrl()
{
return this->pushUrl;
}
FFmpegThread *NetPushClient::getVideoThread()
{
return this->ffmpegThread;
}
void NetPushClient::checkRecord()
{
//0. 时长单位分钟/触发条件自动重新录像/recordDuration=0/表示禁用录像
//1. recordInteger参数控制是否整数倍数录像/recordDuration参数控制录制文件时长/整数倍录像下时长为对应的模数
//2. 一般监控行业会按照整点录像/比如30分钟60分钟一个视频文件/这样录制的文件起始时间和结束时间整整齐齐
//3. 整点录像情况下除了第一个和最后一个录像文件可能时长不一样/中间的文件肯定时长都一样
//4. 非整点录像就按照录像总时长计时/所有保存的文件都是按照时长保存的
//5. recordInteger=true/recordDuration=5/表示每到5分钟的时候录制一个文件
//6. 上面录制结果: 11:01开始录制/11:05结束上一个录制并重新录制/第一个文件时长4分钟
//7. recordInteger=false/recordDuration=5/表示每过5分钟的时候录制一个文件
//8. 上面录制结果: 11:01开始录制/11:06结束上一个录制并重新录制/第一个文件时长5分钟
bool ok = false;
QDateTime now = QDateTime::currentDateTime();
qint64 offset = recordTime.msecsTo(now);
if (recordInteger && recordDuration > 1) {
QTime time = now.time();
int min = time.minute();
int sec = time.second();
min = (min == 0 ? 60 : min);
ok = ((min % recordDuration == 0) && sec >= 0 && sec <= 2);
//qDebug() << TIMEMS << min << sec << (min % recordDuration == 0) << offset << ok;
} else {
ok = (offset >= (recordDuration * 60 * 1000));
//qDebug() << TIMEMS << recordDuration << offset << ok;
}
if (ok && offset >= 5000) {
this->record();
}
}
void NetPushClient::record()
{
if (ffmpegSave) {
//取出推流码
QString flag = pushUrl.split("/").last();
//文件名不能包含特殊字符/需要替换成固定字母
QString pattern("[\\\\/:|*?\"<>]|[cC][oO][mM][1-9]|[lL][pP][tT][1-9]|[cC][oO][nM]|[pP][rR][nN]|[aA][uU][xX]|[nN][uU][lL]");
#if (QT_VERSION >= QT_VERSION_CHECK(6,0,0))
QRegularExpression rx(pattern);
#else
QRegExp rx(pattern);
#endif
flag.replace(rx, "X");
//文件名加上时间结尾
QString path = QString("%1/video/%2").arg(qApp->applicationDirPath()).arg(QDATE);
QString name = QString("%1/%2_%3.mp4").arg(path).arg(flag).arg(STRDATETIME);
//目录不存在则新建
QDir dir(path);
if (!dir.exists()) {
dir.mkpath(path);
}
//先停止再打开重新录制
ffmpegSave->stop();
ffmpegSave->open(name);
recordTime = QDateTime::currentDateTime();
}
}
void NetPushClient::receivePlayStart(int time)
{
//演示添加OSD后推流
#ifdef betaversion
int height = ffmpegThread->getVideoHeight();
QList<OsdInfo> osds = OsdGraph::getTestOsd(height);
ffmpegThread->setOsdInfo(osds);
#endif
//打开后才能启动录像
ffmpegThread->recordStart(pushUrl);
//推流以外还单独存储
if (!ffmpegSave && recordDuration > 0) {
//源头保存没成功就不用继续
FFmpegSave *saveFile = ffmpegThread->getSaveFile();
if (!saveFile->getIsOk()) {
return;
}
ffmpegSave = new FFmpegSave(this);
//重新编码过的则取视频保存类的对象
AVStream *videoStreamIn = saveFile->getVideoEncode() ? saveFile->getVideoStream() : ffmpegThread->getVideoStream();
AVStream *audioStreamIn = saveFile->getAudioEncode() ? saveFile->getAudioStream() : ffmpegThread->getAudioStream();
ffmpegSave->setSavePara(ffmpegThread->getMediaType(), SaveVideoType_Mp4, videoStreamIn, audioStreamIn);
this->record();
timerRecord->start();
}
}
void NetPushClient::receivePacket(AVPacket *packet)
{
if (ffmpegSave && ffmpegSave->getIsOk()) {
ffmpegSave->writePacket2(packet);
}
FFmpegHelper::freePacket(packet);
}
void NetPushClient::recorderStateChanged(const RecorderState &state, const QString &file)
{
int width = 0;
int height = 0;
int videoStatus = 0;
int audioStatus = 0;
if (ffmpegThread) {
width = ffmpegThread->getVideoWidth();
height = ffmpegThread->getVideoHeight();
FFmpegSave *saveFile = ffmpegThread->getSaveFile();
if (saveFile->getIsOk()) {
if (saveFile->getVideoIndexIn() >= 0) {
if (saveFile->getVideoIndexOut() >= 0) {
videoStatus = (saveFile->getVideoEncode() ? 3 : 2);
} else {
videoStatus = 1;
}
}
if (saveFile->getAudioIndexIn() >= 0) {
if (saveFile->getAudioIndexOut() >= 0) {
audioStatus = (saveFile->getAudioEncode() ? 3 : 2);
} else {
audioStatus = 1;
}
}
}
}
//只有处于录制中才表示正常推流开始
bool start = (state == RecorderState_Recording);
emit pushStart(mediaUrl, width, height, videoStatus, audioStatus, start);
}
void NetPushClient::receiveSaveStart()
{
emit pushChanged(mediaUrl, 0);
}
void NetPushClient::receiveSaveFinsh()
{
emit pushChanged(mediaUrl, 1);
}
void NetPushClient::receiveSaveError(int error)
{
emit pushChanged(mediaUrl, 2);
}
void NetPushClient::setMediaUrl(const QString &mediaUrl)
{
this->mediaUrl = mediaUrl;
}
void NetPushClient::setPushUrl(const QString &pushUrl)
{
this->pushUrl = pushUrl;
}
void NetPushClient::start()
{
if (ffmpegThread || mediaUrl.isEmpty() || pushUrl.isEmpty()) {
return;
}
//实例化视频采集线程
ffmpegThread = new FFmpegThread;
//关联播放开始信号用来启动推流
connect(ffmpegThread, SIGNAL(receivePlayStart(int)), this, SLOT(receivePlayStart(int)));
//关联录制信号变化用来判断是否推流成功
connect(ffmpegThread, SIGNAL(recorderStateChanged(RecorderState, QString)), this, SLOT(recorderStateChanged(RecorderState, QString)));
//设置播放地址
ffmpegThread->setMediaUrl(mediaUrl);
//设置视频模式
#ifdef openglx
ffmpegThread->setVideoMode(VideoMode_Opengl);
#else
ffmpegThread->setVideoMode(VideoMode_Painter);
#endif
//设置通信协议(如果是rtsp视频流建议设置tcp)
//ffmpegThread->setTransport("tcp");
//设置硬解码(和推流无关/只是为了加速显示/推流只和硬编码有关)
//ffmpegThread->setHardware("dxva2");
//设置缓存大小(如果分辨率帧率码流很大需要自行加大缓存)
ffmpegThread->setCaching(8192000);
//设置解码策略(推流的地址再拉流建议开启最快速度)
//ffmpegThread->setDecodeType(DecodeType_Fastest);
//设置读取超时时间超时后会自动重连
ffmpegThread->setReadTimeout(10 * 1000);
//设置连接超时时间(0表示一直连)
ffmpegThread->setConnectTimeout(0);
//设置重复播放相当于循环推流
ffmpegThread->setPlayRepeat(true);
//设置默认不播放音频(界面上切换到哪一路就开启)
ffmpegThread->setPlayAudio(false);
//设置默认不预览视频(界面上切换到哪一路就开启)
ffmpegThread->setPushPreview(false);
//设置保存视频类将数据包信号发出来用于保存文件
FFmpegSave *saveFile = ffmpegThread->getSaveFile();
saveFile->setProperty("checkB", checkB);
saveFile->setSendPacket(recordDuration > 0, false);
connect(saveFile, SIGNAL(receivePacket(AVPacket *)), this, SLOT(receivePacket(AVPacket *)));
connect(saveFile, SIGNAL(receiveSaveStart()), this, SLOT(receiveSaveStart()));
connect(saveFile, SIGNAL(receiveSaveFinsh()), this, SLOT(receiveSaveFinsh()));
connect(saveFile, SIGNAL(receiveSaveError(int)), this, SLOT(receiveSaveError(int)));
//如果是本地设备或者桌面录屏要取出其他参数
VideoHelper::initVideoPara(ffmpegThread, mediaUrl, encodeVideoRatio, encodeVideoScale);
//设置视频编码格式/视频压缩比率/视频缩放比例
ffmpegThread->setEncodeVideo((EncodeVideo)encodeVideo);
ffmpegThread->setEncodeVideoRatio(encodeVideoRatio);
ffmpegThread->setEncodeVideoScale(encodeVideoScale);
//启动播放
ffmpegThread->play();
}
void NetPushClient::stop()
{
//停止推流和采集并彻底释放对象
if (ffmpegThread) {
ffmpegThread->recordStop();
ffmpegThread->stop();
ffmpegThread->deleteLater();
ffmpegThread = NULL;
}
//停止录制
if (ffmpegSave) {
timerRecord->stop();
ffmpegSave->stop();
ffmpegSave->deleteLater();
ffmpegSave = NULL;
}
}
相关推荐
- 三种自建KMS激活系统自动激活windows方法
-
第一种:在windows服务器上搭建主要针对vol版本(win7、win10、win20xx、win2012等等)平台:我自己搭建的windows虚拟机,windows2016的操作系统软件:...
- 重装系统被收98元?避开Windows付费陷阱的实用指南
-
重装系统被收98元?避开Windows付费陷阱的实用指南有网友反映,在重装Windows系统后,屏幕突然弹出“激活系统需支付98元服务费”的提示,疑惑自己是不是遭遇了付费陷阱。事实上,微软官方的Wi...
- Windows Server2012远程桌面服务配置和授权激活
-
安装:注意:安装完毕之后需手动重启一下计算机配置终端服务管理工具---远程桌面服务---RD授权诊断程序,查看当前服务器有没有授权授权:运行—>gpedit.msc->计算机配置---管理...
- 新书速览|Windows Server 2022 系统与网站配置实战
-
讲述桌面体验、ServerCore/NanoServer,容器与云系统的配置1本书内容《WindowsServer2022系统与网站配置实战》秉持作者一贯理论兼具实践的写作风格,以新版的Wi...
- Windows激活全攻略:KMS神钥与专业工具的完美结合!
-
对于许多Windows用户来说,系统的激活是一个必经的过程。虽然Windows操作系统在未经激活的状态下也可以使用一段时间,但长期来看,未激活的系统会限制某些功能并频繁提示用户激活。以下是两种流行的激...
- 微软Win9全新激活技术曝光(微软系统激活有什么用)
-
2014-07-0905:46:00作者:徐日俄罗斯Wzor日前披露了更多关于Windows9的最新消息,据悉,Windows9将会在今年秋季亮相,其宣传口号是“想要开始按钮和开始菜单?如你所...
- 快速激活Windows 10/11:CMD命令详细教程
-
#记录我的2024#激活Windows操作系统是确保系统功能和安全更新正常运行的重要步骤。本文将为您分享如何使用命令提示符(CMD)在Windows10和Windows11上进行激活的详细步骤。...
- Wndows 2019 RDS应用发布部署(rds的安装和应用程序的发布)
-
安装前的准备1、需要提供服务器作为应用中心,应用中心的推荐配置如下表所示。规格建议1-10人11-20人21-50人51-100人100+人CPU4核8核16核内存8GB16GB32GB64GB系统盘...
- 解决 Windows 系统激活难题(如何解决windows激活问题)
-
今天,一位朋友给我说,他手头有三台电脑,均同时弹出系统未激活的提示。他对此毫无头绪,便急忙将电脑上出现的激活提示信息一股脑发给了我。我看到其中一台显示的是“Windows10企业版LTSC尚...
- 自建KMS激活服务器(自建kms激活服务器的风险)
-
自建KMS激活服务器Win10和office安装后,都需要激活才可以使用,一般可以输入购买的MAK激活码进行在线激活,也可以通过KMS激活,网上也有很多激活工具,但这些工具一般都含有病毒或木马程序,容...
- 30秒免费激活windows和office亲测有效!
-
“第三方工具有病毒?”“KMS服务器激活总失效?”今天给大家分享一个开源激活工具——MicrosoftActivationScripts(MAS),无需密钥、不装软件,30秒永久激活Window...
- 「操作系统」Windows 10 LTSC 2019 企业版C大集成更新版
-
Windows10LTSC企业版CHIANNET集成更新优化整合多镜像版,CHIANNET,是USBOS超级PE维护盘工具箱作者,长久以来一直默默的更新着,USBOSPE软件,电脑城装机及...
- 一文看懂Windows激活:自查方法+授权类型科普(Win7/Win10通用)
-
一、如何判断Windows是否永久激活?无论是Win7还是Win10,均可通过以下方法快速验证:命令提示符法(通用):按下Win+R,输入slmgr.vbs/xpr并按回车键运行即可查看是否...
- 部分Windows Server 2019/2022用户反馈无法运行微软Teams应用
-
IT之家7月2日消息,科技媒体borncity今天(7月2日)发布博文,报道称在多个WindowsServer版本上,MicrosoftTeams应用近期出现了运行故障。用...
- 这种Windows激活方式已有20年...(windows现在激活)
-
2006年微软正式发布WindowsVista,随之而来引入了一项新的激活机制「OEM激活」,这项机制在Vista和Win7上最为流行。其实WindowsServer自2008开始至2025版本一...
你 发表评论:
欢迎- 一周热门
-
-
UOS服务器操作系统防火墙设置(uos20关闭防火墙)
-
极空间如何无损移机,新Z4 Pro又有哪些升级?极空间Z4 Pro深度体验
-
NAS:DS video/DS file/DS photo等群晖移动端APP远程访问的教程
-
如何在安装前及安装后修改黑群晖的Mac地址和Sn系列号
-
手机如何设置与显示准确时间的详细指南
-
如何修复用户配置文件服务在 WINDOWS 上登录失败的问题
-
一加手机与电脑互传文件的便捷方法FileDash
-
日本海上自卫队的军衔制度(日本海上自卫队的军衔制度是什么)
-
10个免费文件中转服务站,分享文件简单方便,你知道几个?
-
爱折腾的特斯拉车主必看!手把手教你TESLAMATE的备份和恢复
-
- 最近发表
-
- 三种自建KMS激活系统自动激活windows方法
- 重装系统被收98元?避开Windows付费陷阱的实用指南
- Windows Server2012远程桌面服务配置和授权激活
- 新书速览|Windows Server 2022 系统与网站配置实战
- Windows激活全攻略:KMS神钥与专业工具的完美结合!
- 微软Win9全新激活技术曝光(微软系统激活有什么用)
- 快速激活Windows 10/11:CMD命令详细教程
- Wndows 2019 RDS应用发布部署(rds的安装和应用程序的发布)
- 解决 Windows 系统激活难题(如何解决windows激活问题)
- 自建KMS激活服务器(自建kms激活服务器的风险)
- 标签列表
-
- linux 查询端口号 (58)
- docker映射容器目录到宿主机 (66)
- 杀端口 (60)
- yum更换阿里源 (62)
- internet explorer 增强的安全配置已启用 (65)
- linux自动挂载 (56)
- 禁用selinux (55)
- sysv-rc-conf (69)
- ubuntu防火墙状态查看 (64)
- windows server 2022激活密钥 (56)
- 无法与服务器建立安全连接是什么意思 (74)
- 443/80端口被占用怎么解决 (56)
- ping无法访问目标主机怎么解决 (58)
- fdatasync (59)
- 405 not allowed (56)
- 免备案虚拟主机zxhost (55)
- linux根据pid查看进程 (60)
- dhcp工具 (62)
- mysql 1045 (57)
- 宝塔远程工具 (56)
- ssh服务器拒绝了密码 请再试一次 (56)
- ubuntu卸载docker (56)
- linux查看nginx状态 (63)
- tomcat 乱码 (76)
- 2008r2激活序列号 (65)