百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

Qt+Opencv+Ffmpeg实时摄像头数据推流,并在WEB端显示

off999 2025-03-02 18:30 30 浏览 0 评论

0.前言

本文主要介绍下利用Opencv获取摄像头视频数据,并结合ffmpeg推流到流服务器,并在前端页面中显示的整个思路与实现。

1.整体结构

screenviewer服务内部结构分布:

2.方案分析

整体方案如第二部分所示,主要由四部分组成:硬件视频采集设备、视频捕捉模块、流媒体服务、流媒体播放器。涉及的功能技术点有三块:a.实时视频采集与推送;b.rtmp流服务器的搭建;c.流媒体播放器的选择。

a.实时视频采集与推送

采用OpenCV开源视觉库结合FFmpeg进行开发,主要代码片段如下:

初始化采集设备:

if(!m_isCameraData)
{
    qDebug()<< "m_inputSrc: " << m_inputSrc;
    m_cam.open(m_inputSrc.toStdString().data());
}else{
    qDebug()<< "m_videoIndex: " << m_videoIndex;
    m_cam.open(m_videoIndex);
}
 
if (!m_cam.isOpened())
{
    qDebug()<< "camera open failed";
    ret = false;
    return ret;
}
 
// 获取图像属性
m_inWidth = m_cam.get(cv::CAP_PROP_FRAME_WIDTH);
m_inHeight = m_cam.get(cv::CAP_PROP_FRAME_HEIGHT);
m_fps = m_cam.get(cv::CAP_PROP_FPS);

ffmpeg相关初始化:

int VideoWorker::initPushVideo()
{
    int code = 0;
 
    bool ret = false;
 
    // 2.初始化格式转换上下文
    ret = initSwsContext();
    if (!ret)
    {
        qWarning("初始化格式转换上下文失败");
        code = 2;
        return code;
    }
 
    // 3.初始化输出的数据结构
    ret = initOutYuv();
    if (!ret)
    {
        qWarning("初始化输出YUV失败");
        code = 3;
        return code;
    }
 
    // 4.初始化编码上下文
    ret = initVideoCodeC();
    if (!ret)
    {
        qWarning("初始化编码上下文失败");
        code = 4;
        return code;
    }
 
    // 输出封装器和视频流配置
    ret = initOutCAndVideoCfg();
    if (!ret)
    {
        qWarning("输出封装器和视频流配置失败");
        code = 5;
        return code;
    }
 
    return code;
}

各个部分的初始化:

int VideoWorker::initPushVideo()
{
    int code = 0;
 
    bool ret = false;
 
    // 2.初始化格式转换上下文
    ret = initSwsContext();
    if (!ret)
    {
        qWarning("初始化格式转换上下文失败");
        code = 2;
        return code;
    }
 
    // 3.初始化输出的数据结构
    ret = initOutYuv();
    if (!ret)
    {
        qWarning("初始化输出YUV失败");
        code = 3;
        return code;
    }
 
    // 4.初始化编码上下文
    ret = initVideoCodeC();
    if (!ret)
    {
        qWarning("初始化编码上下文失败");
        code = 4;
        return code;
    }
 
    // 输出封装器和视频流配置
    ret = initOutCAndVideoCfg();
    if (!ret)
    {
        qWarning("输出封装器和视频流配置失败");
        code = 5;
        return code;
    }
 
    return code;
}

工作线程:

void VideoWorker::doVideoWork()
{
    bool ret = true;
    while(m_isDoWork)
    {
        // 连续100帧图像未采集到,则认为结束,通知主线程结束任务
        if(m_noframeCount>=100)
        {
            // 通知停止工作
            emit workOver();
        }
        /*读取rtsp视频下一帧,解码视频帧*/
        if (!m_cam.grab())
        {
            m_noframeCount++;
            continue;
        }
        /*yuv转换为rgb*/
        if (!m_cam.retrieve(m_frame))
        {
            continue;
        }
 
        // 采集到图像则计数器清零
        m_noframeCount= 0;
        if(m_isSaveFile)
        {
            m_writer << m_frame;
        }
        if(m_isGray)
        {
            cv::Mat gray;
            cv::cvtColor(m_frame, gray, cv::COLOR_BGR2GRAY);
            emit imageGetted(matToPixmap(gray));
        }else
        {
            emit imageGetted(matToPixmap(m_frame));
        }
        cv::waitKey(m_fps);
        if(!m_isPush)
        {
            continue;
        }
        if(!m_pushCfgReady)
        {
            continue;
        }
 
        /*rgb to yuv*/
        //输入的数据结构
        uint8_t *indata[AV_NUM_DATA_POINTERS] = { 0 };
 
        indata[0] = m_frame.data;
        int insize[AV_NUM_DATA_POINTERS] = { 0 };
 
        //一行(宽)数据的字节数
        insize[0] = m_frame.cols * m_frame.elemSize();
        int h = sws_scale(m_vsc, indata, insize, 0, m_frame.rows, //源数据
                          m_yuv->data, m_yuv->linesize);
        if (h <= 0)
        {
            continue;
        }
 
        /*h264编码*/
        m_yuv->pts = m_vpts;
        m_vpts++;
 
        ret = avcodec_send_frame(m_vc, m_yuv);
        if (ret != 0)
            continue;
 
        ret = avcodec_receive_packet(m_vc, &m_pack);
        if (ret != 0 || m_pack.size > 0)
        {
            // qDebug() << "*" + m_pack.size << flush;
        }
        else
        {
            continue;
        }
 
        //推流
        m_pack.pts = av_rescale_q(m_pack.pts, m_vc->time_base, m_vs->time_base);
        m_pack.dts = av_rescale_q(m_pack.dts, m_vc->time_base, m_vs->time_base);
        m_pack.duration = av_rescale_q(m_pack.duration, m_vc->time_base, m_vs->time_base);
 
        ret = av_interleaved_write_frame(m_ic, &m_pack);
        if (ret == 0)
        {
            Q_UNUSED(ret);
        }
    }
    if(m_cam.isOpened())
    {
        m_cam.release();
    }
    destoryPointers();
}

b.rtmp流服务器的搭建

此部分有大体两个方案:Ⅰ.nginx+rtmp模块,最终会依赖浏览器的flash,现在浏览器基本都放弃了flash,所以我也放弃了这个方案;Ⅱ.nodejs的npm模块,即node-meida-server,不依赖flash,我选择了它。

下载:

npm i node-media-server --save

在node_modules同级目录下,新建app.js,写入以下内容:

/*
 * @Descripttion: 
 * @version: 
 * @Author: zhen
 * @Date: 2022-08-23 21:53:54
 * @LastEditors: 
 * @LastEditTime: 2022-08-23 21:54:01
 */
const NodeMediaServer = require('node-media-server');
 
const config = {
  rtmp: {
    port: 1935,
    chunk_size: 60000,
    gop_cache: true,
    ping: 30,
    ping_timeout: 60
  },
  http: {
    port: 8000,
    allow_origin: '*'
  }
};
 
var nms = new NodeMediaServer(config)
nms.run();

启动:

node app.js

c. 网页端流媒体播放器

找了好多,最终选择了videojs-flvjs。

不依赖flash,美得很!

3.演示

1.先启动nms,即流媒体服务器。

2.启动开发的软件

3.选择推流、点击开始!

web端效果:

注意:上面的视频流服务地址也可以填写其他的,比如B站推流地址!即可在B站直播间看到您的精彩表演!

4.结束

拜拜,欢迎一起交流!

相关推荐

win10版本回退(win10回退到以前版本)

如果你想在Windows10系统中回退到上一个版本,可以按照以下步骤进行操作:1.打开设置:点击Windows开始按钮,然后点击屏幕左侧的“设置”图标,或者使用键盘快捷键Win+I打开设置。2...

营业厅一个路由器多少钱(上门更换路由器收费吗)

移动免费装宽带活动全国都在搞,不过免费是有“门槛”的。以我所在的地区为例,只有月费在78元及以上的大流量套餐用户,才可以享受免费安装移动的宽带。月费越高,宽带的速率也越高,148元档可以安装200M的...

win10从u盘启动怎么设置(win10怎么从u盘启动电脑)

1.回到桌面。点击开始徽标,点击开始菜单左侧的设置。2.设置界面点击更新和安全。3.进入更新和安全界面,点击左侧的恢复选项。4.进入恢复界面,点击高级启动下面的立即重新启动。5.插入自己的U盘,等待...

系统大全网站(系统大全网站推荐)

下载时发生错误可能是以下原因:1.你的网速过慢,网页代码没有完全下载就运行了,导致不完整,当然就错误了。请刷新。2.网页设计错误,导致部分代码不能执行。请下载最新的遨游浏览器。3.你的浏览器不兼容导致...

win10官方启动盘(win10官方启动盘怎么用)

1、在开始菜单搜索“设置”,打开“设置”;2、点击“更新与安全”,在左侧菜单栏点击“恢复”;3、点击“启动项”,在弹出的窗口中会显示当前可以启动的项目,点击“编辑”;4、在打开的“编辑启动项”窗口中,...

win10系统安装不了(win10 安装不了)

电脑装不上win10系统可能是因为以下几个原因导致的原因一:win10安装文件不对我们在安装win10之前,要确保下载到安装包真实可用的,否则安装肯定会有问题,建议下载安全可靠的安装包!原因二:系统文...

国内dns哪个最快(dns开启好还是关闭好)

移动dns设置首选114.114.114.114,它又好又快。首选DNS和备用DNS都是一种域名系统,这两种域名系统有着先后之分,如果在首选DNS正常的情况下,就用首选DNS地址。当首选DNS服务器出...

winxp安装盘(winxp系统安装)

xp系统安装步骤如下1、将下载的xp系统iso压缩包文件下载到C盘之外的分区,比如下载到D盘,右键使用WinRAR等工具解压到当前文件夹或指定文件夹,不能解压到C盘和桌面,否则无法安装;?2、解压之后...

现在的win11稳定了吗(win11稳定嘛)

windows10更稳定,由于win11刚刚推出没多久,稳定差不够好,兼容性也有待提升,无论是应用还是游戏都会遇到不明程度的问题,因此,在日常的使用过程中,我们还是应当以稳定性为优先,选择win10是...

xp安装包下载到手机(xp系统安装包)

手机是基于ARM架构的处理器,而WindowsXP是基于x86架构的操作系统,因此无法直接在手机上安装WindowsXP。除非您的手机是使用Intel处理器,但这种情况非常罕见。如果您需要在手机上...

如何查看硬盘序列号(windows如何查看硬盘序列号)

1.打开开始菜单栏,输入【cmd】点击【确定】;2.在命令窗口依次输入【diskpart】-【listdisk】-【selectdisk0】;3.选好要查看的硬盘后,接着输入【detaildi...

虚拟机安装win7教程(虚拟机安装win7教程图解)

1.首先,下载并安装虚拟机软件,如VMwareWorkstation、VirtualBox等。2.打开虚拟机软件,创建一个新的虚拟机。3.在创建虚拟机的过程中,选择安装Windows7专业版的IS...

系统脱敏法的操作程序如何

系统脱敏疗法(systematicdesensitization)又称交互抑制法,是由美国学者沃尔普创立和发展的。这种方法主要是诱导求治者缓慢地暴露出导致神经症焦虑、恐惧的情境,并通过心理的放松状态...

闪迪u盘低级格式化工具(闪迪u盘格式化分配单元大小)

闪迪U盘格式化后速度变慢的可能原因及解决方法如下:文件系统问题:格式化时选择的文件系统类型可能会影响U盘的性能。常见的文件系统类型包括FAT32、NTFS和exFAT等。如果文件系统类型不合适,可能会...

psd文件下载(psd格式下载网站)

  1、在photoshop中,不能通过置入的方法来加载PSD文件,因为,通过置入的方法加载PSD文件,它是以合并图层的方法把PSD文件加入,这样,就失去了PSD文件的所有图层信息。  2、在文档中想...

取消回复欢迎 发表评论: