Qt/C++摄像头采集/二维码解析/同时采集多路/图片传输/分辨率帧率可调/自动重连

一、前言

本地摄像头的采集可以有多种方式,一般本地摄像头会通过USB的方式连接,在嵌入式上可能大部分是CMOS之类的软带的接口,这些都统称本地摄像头,和网络摄像头最大区别就是一个是通过网络来通信,一个是直接本地通信。本地摄像机的采集可以用qcamera来做,但是qcamera类在很多平台没有实现,比如嵌入式linux系统上几乎是没有用的,所以更加推荐用兼容性跨平台最好的ffmpeg来做,在linux上当然也可以通过v4l2来实现,这个其实是整个linux系统中通用的USB摄像头采集的框架,无论是ffmpeg还是qcamera,在linux系统上底层其他都是用v4l2来实现的,所以如果在嵌入式板子受限于内存或者存储空间大小,用不了ffmpeg一堆库,这个时候就可以用最底层的v4l2来采集。

很多人的应用场景要求不止采集1路本地摄像头,可能有多路都需要同时采集,其实这个和程序没有太大关系,能够采集一路肯定就能多路,毕竟就是封装的一个类,直接new出来就行,要能采集多路最大的性能瓶颈在USB带宽,数据带宽,还有就是USB口子的供电足不足,供电不足,也只能采集1-2路,比如有些人用的是USB集线器,上面一排USB口子,由于供电不足,就算你插4个USB摄像头,也只能最多采集2路,带宽又限制了分辨率和帧率,比如一个普通PC机器可以同时采集4路640x480分辨率,但是由于带宽不足,无法同时采集4路1080P,尽管4个摄像头都支持1080p分辨率,所以这是个综合因素的叠加影响,遇到问题慢慢查。

这个摄像头综合应用,陆陆续续完善了很多年,最开始是v4l2版本,只能用在嵌入式linux上,而后从Qt5开始集成了qcamera类,所以有用qcamera来做了一个版本,现在Qt6在多媒体框架有了巨大的性能提升,又用Qt6的qcamera做了个版本,为什么不通用了?因为Qt6的多媒体框架做了巨大更新调整,完全不兼容之前的类。在做视频组件的实现用ffmpeg采集本地摄像头也实现了,而且做的比较完善,所以后面单独提炼出来做了ffmpeg的版本,最后测试下来发现ffmpeg是最通用的,在哪里都能正常采集。

做二维码采集这个功能,用的是zxing类,最初的做法是将采集的图片发给zxing解析,实际过程发现大图片解析很慢,比如1080P图片大概1s只能解析3张,这个速度看起来就慢了点,如果是2K的分辨率呢。后面经过一些真实场景应用,包括菜鸟驿站里面带的扫描识别,发现都有个放大区域,用户可以设置一个关心的矩形区域,要放在这个矩形区域内,能够最大最快的识别,这是个不错的策略,在ffmpeg中通过设置crop裁剪区域就可以直接将对应内容放大显示,然后采集的实时画面可以是原视频,发送给zxing解析的就是裁剪后的画面,这样速度大大提升。之前大图就算二维码在里面,由于在整个图片中的占比区域太小,也可能解析失败,自动调整后裁剪区域,性能大幅度提升,完美解决。

二、效果图

在这里插入图片描述

三、体验地址

  1. 国内站点:https://gitee.com/feiyangqingyun
  2. 国际站点:https://github.com/feiyangqingyun
  3. 个人作品:https://blog.csdn.net/feiyangqingyun/article/details/97565652
  4. 体验地址:https://pan.baidu.com/s/1d7TH_GEYl5nOecuNlWJJ7g 提取码:01jf 文件名:bin_video_camera。

四、相关代码

#include "frmzxing.h"
#include "frmmain.h"
#include "ui_frmzxing.h"
#include "qthelper.h"
#include "camerahelper.h"
#include "widgethelper.h"
#include "zxingthread.h"frmZxing::frmZxing(QWidget *parent) : QWidget(parent), ui(new Ui::frmZxing)
{ui->setupUi(this);this->initForm();this->initConfig();
}frmZxing::~frmZxing()
{delete ui;
}void frmZxing::initForm()
{ui->frame->setFixedWidth(AppData::RightWidth);//关联信号槽connect(ui->cameraWidget, SIGNAL(sig_receivePlayStart(int)), this, SLOT(receivePlayStart(int)));connect(ui->cameraWidget, SIGNAL(sig_receivePoint(int, QPoint)), this, SLOT(receivePoint(int, QPoint)));//实例化解析类并启动线程zxing = new ZXingThread(this);connect(zxing, SIGNAL(receiveResult(QString, QString, int)), this, SLOT(receiveResult(QString, QString, int)));zxing->start();//启动定时器截图timer = new QTimer(this);connect(timer, SIGNAL(timeout()), this, SLOT(on_btnSnap_clicked()));if (AppConfig::Zxing_Interval > 0) {timer->start(AppConfig::Zxing_Interval);}
}void frmZxing::initConfig()
{CameraHelper::loadCameraCore(ui->cboxCameraCore, AppConfig::Zxing_CameraCore);connect(ui->cboxCameraCore, SIGNAL(currentIndexChanged(int)), this, SLOT(saveConfig()));this->cameraDeviceChanged();connect(ui->cboxCameraName->lineEdit(), SIGNAL(textChanged(QString)), this, SLOT(saveConfig()));CameraHelper::loadVideoSize(ui->cboxVideoSize);ui->cboxVideoSize->lineEdit()->setText(AppConfig::Zxing_VideoSize);connect(ui->cboxVideoSize->lineEdit(), SIGNAL(textChanged(QString)), this, SLOT(saveConfig()));CameraHelper::loadFrameRate(ui->cboxFrameRate);ui->cboxFrameRate->lineEdit()->setText(QString::number(AppConfig::Zxing_FrameRate));connect(ui->cboxFrameRate->lineEdit(), SIGNAL(textChanged(QString)), this, SLOT(saveConfig()));ui->cboxVideoMode->setCurrentIndex(AppConfig::Zxing_VideoMode);connect(ui->cboxVideoMode, SIGNAL(currentIndexChanged(int)), this, SLOT(saveConfig()));zxing->setDecoder(AppConfig::Zxing_DecoderFormat);ui->cboxDecoderFormat->addItem("仅二维码");ui->cboxDecoderFormat->addItem("一维二维");ui->cboxDecoderFormat->addItem("所有格式");ui->cboxDecoderFormat->setCurrentIndex(AppConfig::Zxing_DecoderFormat);connect(ui->cboxDecoderFormat, SIGNAL(currentIndexChanged(int)), this, SLOT(saveConfig()));ui->cboxInterval->addItem("暂停");ui->cboxInterval->addItem("300");ui->cboxInterval->addItem("500");ui->cboxInterval->addItem("800");ui->cboxInterval->addItem("1000");int index = ui->cboxInterval->findText(QString::number(AppConfig::Zxing_Interval));ui->cboxInterval->setCurrentIndex(index < 0 ? 0 : index);connect(ui->cboxInterval, SIGNAL(currentIndexChanged(int)), this, SLOT(saveConfig()));ui->txtTopLeft->setText(AppConfig::Zxing_TopLeft);connect(ui->txtTopLeft, SIGNAL(textChanged(QString)), this, SLOT(saveConfig()));ui->txtBottomRight->setText(AppConfig::Zxing_BottomRight);connect(ui->txtBottomRight, SIGNAL(textChanged(QString)), this, SLOT(saveConfig()));
}void frmZxing::saveConfig()
{//内核变了需要重新搜索设备int cameraCore = ui->cboxCameraCore->itemData(ui->cboxCameraCore->currentIndex()).toInt();if (AppConfig::Zxing_CameraCore != cameraCore) {AppConfig::Zxing_CameraCore = cameraCore;this->cameraDeviceChanged();}//不为空才需要改变QString cameraName = ui->cboxCameraName->currentText();if (!cameraName.isEmpty()) {AppConfig::Zxing_CameraName = cameraName;}AppConfig::Zxing_VideoSize = ui->cboxVideoSize->currentText();AppConfig::Zxing_FrameRate = ui->cboxFrameRate->currentText().toInt();AppConfig::Zxing_VideoMode = ui->cboxVideoMode->currentIndex();//变了立即重新设置int decoderFormat = ui->cboxDecoderFormat->currentIndex();if (AppConfig::Zxing_DecoderFormat != decoderFormat) {AppConfig::Zxing_DecoderFormat = decoderFormat;zxing->setDecoder(AppConfig::Zxing_DecoderFormat);}int interval = ui->cboxInterval->currentText().toInt();if (AppConfig::Zxing_Interval != interval) {AppConfig::Zxing_Interval = interval;if (interval > 0) {timer->start(interval);} else {timer->stop();}}AppConfig::Zxing_TopLeft = ui->txtTopLeft->text().trimmed();AppConfig::Zxing_BottomRight = ui->txtBottomRight->text().trimmed();AppConfig::writeConfig();
}void frmZxing::initPara()
{//设置显示窗体参数WidgetPara widgetPara = ui->cameraWidget->getWidgetPara();widgetPara.borderWidth = 1;widgetPara.videoMode = (VideoMode)AppConfig::Zxing_VideoMode;ui->cameraWidget->setWidgetPara(widgetPara);//设置采集线程参数CameraPara cameraPara = ui->cameraWidget->getCameraPara();cameraPara.cameraCore = (CameraCore)AppConfig::Zxing_CameraCore;cameraPara.cameraName = AppConfig::Zxing_CameraName;cameraPara.videoSize = AppConfig::Zxing_VideoSize;cameraPara.frameRate = AppConfig::Zxing_FrameRate;ui->cameraWidget->setCameraPara(cameraPara);
}void frmZxing::append(int type, const QString &data, bool clear)
{static int maxCount = 50;static int currentCount = 0;QtHelper::appendMsg(ui->txtMain, type, data, maxCount, currentCount, clear);ui->txtMain->moveCursor(QTextCursor::End);
}void frmZxing::createImage(const QString &text)
{if (!text.isEmpty()) {QImage image = zxing->encodeData(text, QSize(250, 250));ui->labResult->setImage(image, true);}
}void frmZxing::cameraDeviceChanged()
{frmMain::getCameraInfo((CameraCore)AppConfig::Zxing_CameraCore);CameraHelper::loadCameraName(ui->cboxCameraName, AppConfig::Zxing_CameraName);
}void frmZxing::snapImage(const QImage &image, const QString &snapName)
{QImage img = image;if (snapName != "file") {//有左上右下坐标说明需要裁减int w = image.width();int h = image.height();if (!AppConfig::Zxing_TopLeft.isEmpty() && !AppConfig::Zxing_BottomRight.isEmpty()) {QRect rect = WidgetHelper::getRect(AppConfig::Zxing_TopLeft, AppConfig::Zxing_BottomRight);img = img.copy(rect);QString msg = QString("执行裁剪(原分辨率: %1 x %2  新分辨率: %3 x %4)").arg(w).arg(h).arg(img.width()).arg(img.height());append(4, msg);img.save("f:/1.jpg", "jpg");}//防止图片过大导致解析很慢static int maxWidth = 1280;static int maxHeight = 720;w = img.width();h = img.height();if (w > maxWidth || h > maxHeight) {img = img.scaled(QSize(maxWidth, maxHeight), Qt::KeepAspectRatio);QString msg = QString("执行缩放(原分辨率: %1 x %2  新分辨率: %3 x %4)").arg(w).arg(h).arg(maxWidth).arg(maxHeight);append(4, msg);}}//添加到线程中解析zxing->append("", img);//显示图片ui->labImage->setImage(img, true);
}void frmZxing::receiveResult(const QString &flag, const QString &text, int time)
{append(2, QString("(用时: %1 毫秒) 结果: %2").arg(time).arg(text));ui->txtResult->setText(text);//重新生成新的二维码图片createImage(text);
}void frmZxing::receivePlayStart(int time)
{//如果存在裁剪区域则设置图形if (!AppConfig::Zxing_TopLeft.isEmpty() && !AppConfig::Zxing_BottomRight.isEmpty()) {GraphInfo graph;graph.rect = WidgetHelper::getRect(AppConfig::Zxing_TopLeft, AppConfig::Zxing_BottomRight);graph.borderWidth = WidgetHelper::getBorderWidth(ui->cameraWidget);ui->cameraWidget->setGraph(graph);}
}void frmZxing::receivePoint(int type, const QPoint &point)
{QString text = QString("%1, %2").arg(point.x()).arg(point.y());if (ui->rbtnTopLeft->isChecked()) {ui->txtTopLeft->setText(text);} else {ui->txtBottomRight->setText(text);}
}void frmZxing::on_btnPlay_clicked()
{this->initPara();if (!ui->cameraWidget->init()) {return;}//关联采集线程信号槽CameraThread *cameraThread = ui->cameraWidget->getCameraThread();connect(cameraThread, SIGNAL(cameraDeviceChanged()), this, SLOT(cameraDeviceChanged()));connect(cameraThread, SIGNAL(snapImage(QImage, QString)), this, SLOT(snapImage(QImage, QString)));ui->cameraWidget->play();ui->widget->setEnabled(false);
}void frmZxing::on_btnStop_clicked()
{ui->cameraWidget->stop();ui->widget->setEnabled(true);
}void frmZxing::on_btnPause_clicked()
{ui->cameraWidget->pause();
}void frmZxing::on_btnNext_clicked()
{ui->cameraWidget->next();
}void frmZxing::on_btnSnap_clicked()
{CameraThread *thread = ui->cameraWidget->getCameraThread();if (thread && thread->getIsOk()) {append(4, "执行截图并解析二维码");ui->cameraWidget->snap();}
}void frmZxing::on_btnFile_clicked()
{QString fileName = QFileDialog::getOpenFileName(this, "打开", "", "*.jpg *.png");if (!fileName.isEmpty()) {QImage image(fileName);snapImage(image, "file");}
}void frmZxing::on_btnCreate_clicked()
{createImage(ui->txtResult->text());
}

五、功能特点

  1. 同时支持 qcamera、ffmpeg、v4l2 三种内核解析本地摄像头。
  2. 提供函数 findCamera 自动搜索环境中的所有本地摄像头设备,搜索结果信号发出。
  3. 支持自动搜索和指定设备两种模式,自动搜索模式下会将搜索到的第一个设备作为当前设备打开。
  4. 支持同时打开多路设备,亲测4路,受限于具体的环境比如带宽。
  5. 支持自动重连,默认开启,失败后会自动重新搜索和尝试打开。
  6. ffmpeg方案、v4l2方案都支持回调模式(采集后转成QImage绘制)和句柄模式(采集后YUV数据GPU绘制,性能高)。
  7. 视频显示位置自动调整算法,当视频分辨率超过显示控件大小则等比例缩放居中显示,不超过则原尺寸居中显示,还可设置拉伸填充显示。(自动调整、等比例缩放、拉伸填充)。
  8. 可选不同的分辨率来打开摄像头,支持 160x120、320x240、640x480、800x600、1280x720、1280x960、1920x1080 等。
  9. 可选不同的帧率来打开摄像头,支持 0(采用默认值)、5、、10、15、20、25、30 等。
  10. 支持抓拍截图,传入文件名则自动保存截图文件,不传入则将图片数据QImage信号发出。
  11. 提供函数接口 开始播放play、停止播放stop、暂停播放pause、继续播放next。
  12. 支持动态热插拔加载,包括自动读取所有设备名称到下拉框。
  13. 支持录像文件存储,提供开始录像recordStart、暂停录像recordPause、停止录像recordStop 等函数。
  14. 提供二维码示例,自动采集画面识别二维码,支持自动将识别到的二维码重新生成大图。
  15. 二维码识别支持设置热点区域,对该区域内的图片进行裁切并识别,在大分辨率图像采集的时候非常有用,提升速度和效率。
  16. 支持选择图片文件解析二维码,手动输入文本内容生成二维码。
  17. 提供图片传输示例,自动将打开的摄像头视频实时传输出去,服务器端接收后解析显示。此方案可以作为将本地的摄像头实时画面远程传输,比如嵌入式板子上的摄像头画面传输到PC端显示。
  18. 支持等比例拉伸填充显示,画面宽高小于显示控件的宽高则以原视频大小为准,大于则按照显示控件的尺寸等比例缩放居中。
  19. 视频控件悬浮条自带开始和停止录像切换、声音静音切换、抓拍截图、关闭视频等功能。
  20. 音频组件支持声音波形值数据解析,可以根据该值绘制波形曲线和柱状声音条,默认提供了声音振幅信号。
  21. 代码框架和结构优化到极致,性能彪悍,持续迭代更新升级。
  22. 源码支持Qt4、Qt5、Qt6,兼容所有版本。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/605881.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

浅谈顺序表基本操作

&#x1f937;‍♀️&#x1f937;‍♀️&#x1f937;‍♀️ 今天给大家带来的是数据结构——顺序表的实现(增删查改)。 清风的CSDN博客主页 &#x1f389;欢迎&#x1f44d;点赞✍评论❤️收藏 &#x1f61b;&#x1f61b;&#x1f61b;希望我的文章能对你有所帮助&#xff…

JavaWeb基础(2)- Web概述、HTTP协议、Servlet、Request与Response

JavaWeb基础&#xff08;2&#xff09;- Web概述、HTTP协议、Servlet、Request与Response 文章目录 JavaWeb基础&#xff08;2&#xff09;- Web概述、HTTP协议、Servlet、Request与Response3 Web概述3.1 Web和JavaWeb的概念3.2 JavaWeb技术栈3.2.1 B/S架构**3.2.2 静态资源**3…

不带控制器打包exe,转pdf文件时失败的原因

加了注释的两条代码后&#xff0c;控制器会显示一个docx转pdf的进度条。这个进度条需要控制器的实现&#xff0c;如果转exe不带控制器的话&#xff0c;当点击转换为pdf的按钮就会导致程序出错和闪退。 __init__.py文件的入口

Netplan介绍

1 介绍 1.1 简介 Netplan是一个抽象网络配置描述器。通过netplan命令&#xff0c;你只需用一个 YAML文件描述每个网络接口所需配置。netplan并不关系底层管理工具是NetworkManager还是networkd。 它是一个在 Linux 系统上进行网络配置的实用程序。您创建所需接口的描述并定义…

java: 5-3 for

文章目录 1. for1.1 基本语法1.2 练习1.3 执行流程1.4 细节1.5 编程思想 (练习) 1. for 1.1 基本语法 for 关键字&#xff0c;表示循环控制for 有四要素: (1)循环变量初始化(2)循环条件(3)循环操作(4)循环变量迭代循环操作 , 这里可以有多条语句&#xff0c;也就是我们要循环…

FreeRTOS学习第6篇–任务状态挂起恢复删除等操作

目录 FreeRTOS学习第6篇--任务状态挂起恢复删除等操作任务的状态设计实验IRReceiver_Task任务相关代码片段实验现象本文中使用的测试工程 FreeRTOS学习第6篇–任务状态挂起恢复删除等操作 本文目标&#xff1a;学习与使用FreeRTOS中的几项操作&#xff0c;有挂起恢复删除等操作…

在MeshLab中创建简单的几何对象

文章目录 立方体和平面网格正多面体圆形相关球类隐式曲面 在Filters->Create New Mesh Layer的子菜单中&#xff0c;提供了大量几何对象&#xff0c;列表如下 菜单指令图形菜单指令图形Dodecahedron正十二面体Icosahedron正二十面体Tetrahedron正四面体Octahedron正八面体B…

Kafka(五)生产者

目录 Kafka生产者1 配置生产者bootstrap.serverskey.serializervalue.serializerclient.id""acksallbuffer.memory33554432(32MB)compression.typenonebatch.size16384(16KB)max.in.flight.requests.per.connection5max.request.size1048576(1MB)receive.buffer.byte…

xdoj托普利兹矩阵

#include <stdio.h> int main() {char Hn0,Cn0;int i0,n,j,h[10],c[10],a[10][10];while(Hn!\n)//输入 行向量{scanf("%d",&h[i]);i;scanf("%c",&Hn);}i0;while(Cn!\n)//输入 列向量{scanf("%d",&c[i]);i;scanf("%c&quo…

目标检测中的常见指标

概念引入&#xff1a; TP&#xff1a;True Positive IoU > 阈值 检测框数量 FP: False Positive IoU < 阈值 检测框数量 FN: False Negative 漏检框数量 Precision:查准率 Recall:查全率&#xff08;召回率&#xff09; AP&am…

【精通C语言】:分支结构switch语句的灵活运用

&#x1f3a5; 屿小夏 &#xff1a; 个人主页 &#x1f525;个人专栏 &#xff1a; C语言详解 &#x1f304; 莫道桑榆晚&#xff0c;为霞尚满天&#xff01; 文章目录 &#x1f4d1;前言一、switch语句1.1 语法1.2 代码示例 二、switch的控制语句2.1 break2.2 defualt子句 三、…

【中小型企业网络实战案例 八】配置映射内网服务器和公网多出口、业务测试和保存配置

相关学习文章&#xff1a; 【中小型企业网络实战案例 一】规划、需求和基本配置 【中小型企业网络实战案例 二】配置网络互连互通【中小型企业网络实战案例 三】配置DHCP动态分配地址 【中小型企业网络实战案例 四】配置OSPF动态路由协议【中小型企业网络实战案例 五】配置可…

H266/VVC网络适配层概述

视频编码标准的分层结构 视频数据分层的必要性&#xff1a;网络类型的多样性、不同的应用场景对视频有不同的需求。 编码标准的分层结构&#xff1a;为了适应不同网络和应用需求&#xff0c;视频编码数据根据其内容特性被分成若干NAL单元&#xff08;NAL Unit&#xff0c;NALU…

2024--Django平台开发-基础信息(一)

一、前置知识点 - Python环境搭建 (Python解释器、Pycharm、环境变量等) - 基础语法(条件、循环、输入输出、编码等) - 数据类型(整型、布尔型、字符串、列表、字典、元组、集合等) - 函数(文件操作、返回值、参数、作用域等) - 面向对象 (类、对象、封装、继承、多态等)包和模…

【动态规划】C++算法312 戳气球

作者推荐 【动态规划】【字符串】扰乱字符串 本文涉及的基础知识点 动态规划 LeetCode312 戳气球 有 n 个气球&#xff0c;编号为0 到 n - 1&#xff0c;每个气球上都标有一个数字&#xff0c;这些数字存在数组 nums 中。 现在要求你戳破所有的气球。戳破第 i 个气球&…

Simpy简介:python仿真模拟库-01/5

一、说明 在计算机编程领域&#xff0c;仿真在理解复杂系统、进行实验和做出明智决策方面发挥着关键作用。SimPy 是“Simulation Python”的缩写&#xff0c;是一个功能强大且多功能的仿真框架&#xff0c;允许开发人员和研究人员使用 Python 创建和分析离散事件仿真。无论您是…

视频如何制作微信表情?仅需一招在线制作

Gif动画表情包是当下一种非常流行的图片展示格式&#xff0c;能够通过gif格式的图片来调节聊天氛围或是传递信息&#xff0c;非常有趣。而gif动图现在也被各行各业的商家用作宣传使用&#xff0c;很吸引大众的目光。 那么&#xff0c;这种非常吸引人的gif动图是怎么制作的呢&a…

Power BI - 5分钟学习修改数据类型

每天5分钟&#xff0c;今天介绍Power BI修改数据类型 Power BI加载数据时&#xff0c;会尝试将源列的数据类型转换为更高效的存储、计算和数据可视化的数据类型。 例如&#xff0c;如果从Excel导入的值的列没有小数值&#xff0c;Power BI Desktop会将整个数据列转换为整数数据…

Spanner on a modern columnar storage engine 中文翻译

文章目录 0. 摘要1. 存储引擎2. 存储引擎迁移的挑战2.1 可靠性、可用性和数据完整性2.2 性能和成本2.3 复杂性 3. 迁移可靠性的系统原则方法3.1 可靠性原则和自动化架构3.2 迁移方案和按周迁移3.3 客户 部署感知 调度3.4 管理可靠性、可用性和性能 4. 项目管理和驱动指标概括 0…

基于领域驱动设计的低代码平台的设计与实现

本文介绍了基于领域驱动设计&#xff08;DDD&#xff09;的低代码平台的设计与实现方法。低代码平台是一种能够通过图形化界面和少量编码&#xff0c;快速构建应用程序的工具。通过结合DDD的思想&#xff0c;我们可以将领域专家的知识转化为具体的领域模型&#xff0c;并将其作…