基于Qt/C++/Opencv实现的一个视频中二维码解析软件

本文详细讲解了如何利用 Qt 和 OpenCV 实现一个可从视频和图片中检测二维码的软件。代码实现了视频解码、多线程处理和界面更新等功能,是一个典型的跨线程图像处理项目。以下分模块对代码进行解析。


在这里插入图片描述
在这里插入图片描述

一、项目的整体结构

项目分为以下几部分:

  1. 主窗口 (MainWindow) :负责界面的加载、初始化和用户交互。
  2. 工作线程 (mThread):处理耗时的图像处理任务(如二维码识别)。
  3. 二维码检测逻辑:使用 OpenCV 进行二维码检测,支持图片和视频两种数据来源。
  4. 多线程通信:通过信号与槽机制,在主线程和工作线程之间传递状态与数据。

二、主窗口功能解析

1. 初始化界面和变量

MainWindow 类的构造函数调用了 initializeUI()initializeVariable(),分别完成了界面的样式加载和核心变量的初始化。

void MainWindow::initializeVariable()
{
    m_tip = nullptr;

    m_lamp[0] = QImage(":/Img/e.png");
    m_lamp[1] = QImage(":/Img/i.png");
    m_lamp[2] = QImage(":/Img/w.png");

    mthread = new mThread();  // 创建工作线程
    m_Threadrun = false;

    // 线程信号与主窗口槽函数的连接
    connect(mthread, SIGNAL(RuningState(bool)), this, SLOT(onRespondThreadRuningState(bool)));
    connect(mthread, SIGNAL(errors(QString)), this, SLOT(onRespondThreaderrors(QString)));
    connect(mthread, SIGNAL(infors(QString)), this, SLOT(onRespondThreadinfors(QString)));
    connect(mthread, SIGNAL(warings(QString)), this, SLOT(onRespondThreadwarings(QString)));
    connect(mthread, &mThread::imageProcessed, this, &MainWindow::processImage);
}
2. 启动和停止线程

用户点击按钮后,调用 on_btn_Start_Stop_clicked,判断当前线程状态以启动或停止工作线程。

void MainWindow::on_btn_Start_Stop_clicked()
{
    m_Threadrun ? mthread->stop() : mthread->start(); // 根据当前状态启动或停止线程
}
3. 文件选择

QFileDialog 被用来让用户选择视频或图像文件,并将这些参数传递到线程处理。

void MainWindow::on_btn_Loadfile_clicked()
{
    QString fileName = QFileDialog::getOpenFileName(nullptr, tc("选择视频文件"), "", tc("视频文件(*.mp4)"));
    mthread->setFunId(0); // 设置功能 ID:0 表示处理视频
    if (!fileName.isEmpty())
        mthread->setThreadParams(fileName); // 传递参数到线程
}

void MainWindow::on_btn_Loadimages_clicked()
{
    QStringList fileNames = QFileDialog::getOpenFileNames(nullptr, tc("选择图像文件"), "", tc("图片文件(*.jpg *.bmp *.png)"));
    mthread->setFunId(1); // 设置功能 ID:1 表示处理图片
    if (!fileNames.isEmpty())
        mthread->setThreadParams(fileNames);
}

三、工作线程实现

mThread 类继承自 QThread,用于处理耗时的二维码检测任务。其主要功能包括:

  1. 根据功能 ID 分别处理视频或图片
  2. 在每帧中调用 OpenCV 的 QRCodeDetector 进行二维码检测
  3. 通过信号将处理后的图像和数据传递回主线程
1. 核心线程逻辑

线程的运行逻辑集中在 run() 方法中。getFunId() 决定了是处理视频还是图片,分别调用 anayVideo()anayImages()

void mThread::run()
{
    m_isRun = true;
    emit RuningState(true); // 通知主线程:线程开始运行
    emit infors(tc("线程启动"));

    switch (getFunId()) {
    case 0:
        anayVideo(); // 处理视频
        break;
    case 1:
        anayImages(); // 处理图片
        break;
    default:
        break;
    }

    emit RuningState(false); // 通知主线程:线程结束运行
    emit infors(tc("线程退出"));
}
2. 视频处理

anayVideo() 中,使用 OpenCV 的 VideoCapture 解码视频逐帧处理。每一帧调用 delectDecoded() 检测二维码,并通过信号将结果传回主线程。

void mThread::anayVideo()
{
    cv::VideoCapture cap;
    if (!cap.open(m_Params.toString().toLocal8Bit().data()) || !cap.isOpened())
    {
        emit errors(tc("视频未打开"));
        m_isRun = false;
    }
    else
    {
        cv::Mat frame;
        int frameCount = cap.get(cv::CAP_PROP_FRAME_COUNT);
        while ((frameCount--) > 0 && m_isRun) // 帧循环
        {
            cap >> frame; // 读取一帧
            if (frame.empty())
                break;

            QString msg;
            delectDecoded(frame, msg); // 检测二维码
            emit imageProcessed(MatToQImage(frame), msg); // 发射处理信号
            cv::waitKey(50);
        }
        cap.release();
    }
}
3. 图片处理

图片处理逻辑与视频类似,只是直接从文件路径中读取。

void mThread::anayImages()
{
    QStringList files = m_Params.toStringList();
    for (auto file : files)
    {
        cv::Mat frame = cv::imread(file.toStdString().c_str());
        if (frame.empty() && !m_isRun)
            break;

        QString msg;
        delectDecoded(frame, msg);
        emit imageProcessed(MatToQImage(frame), msg); // 发射信号
        cv::waitKey(1000);
    }
}

四、二维码检测实现

1. 使用 OpenCV 进行检测

delectDecoded() 方法中,利用 OpenCV 的 QRCodeDetector 类进行二维码检测和解码,并将结果绘制到图像中。

int mThread::delectDecoded(cv::Mat &image, QString &code)
{
    cv::Mat bbox, rectifiedImage;
    std::string data = qrDecoder.detectAndDecode(image, bbox, rectifiedImage);

    if (data.length() > 0)
    {
        code = QString::fromStdString(data); // 将结果返回
        std::vector<cv::Point> points;
        for (int i = 0; i < bbox.cols; i++)
        {
            points.push_back(cv::Point(static_cast<int>(bbox.at<cv::Point2f>(0, i).x), static_cast<int>(bbox.at<cv::Point2f>(0, i).y)));
        }
        for (size_t i = 0; i < points.size(); i++)
        {
            cv::line(image, points[i], points[(i + 1) % points.size()], cv::Scalar(0, 255, 0), 3); // 绘制绿色边框
        }
        int minY = points[0].y;
        for (const auto &point : points) {
            minY = std::min(minY, point.y);
        }
        cv::putText(image, data, cv::Point(points[0].x, minY - 10), cv::FONT_HERSHEY_SIMPLEX, 0.7, cv::Scalar(0, 255, 0), 2); // 显示二维码信息
    }
    else
    {
        code = tc("未检测到二维码!");
    }

    return 0;
}
2. Mat 转 QImage

为了在 Qt 界面中显示 OpenCV 的图像,MatToQImage() 将 OpenCV 的 cv::Mat 转换为 Qt 的 QImage


五、多线程与信号槽

在本项目中,多线程通过信号与槽实现以下功能:

  1. 更新主界面状态:线程的运行状态(如启动和停止)通过 RuningState 信号通知主线程。
  2. 实时更新图像和检测结果imageProcessed 信号传递处理后的图像和二维码信息,更新界面。
connect(mthread, &mThread::imageProcessed, this, &MainWindow::processImage);

void MainWindow::processImage(const QImage &image, const QString &msg)
{
    ui->lab_disp->setPixmap(QPixmap::fromImage(image).scaled(image.width() / 2, image.height() / 2)); // 显示缩放后的图像
    ui->lab_disData->setText(msg); // 显示检测到的信息
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【Elasticsearch入门到落地】2、正向索引和倒排索引

接上篇《1、初识Elasticsearch》 上一篇我们学习了什么是Elasticsearch&#xff0c;以及Elastic stack(ELK)技术栈介绍。本篇我们来什么是正向索引和倒排索引&#xff0c;这是了解Elasticsearch底层架构的核心。 上一篇我们学习到&#xff0c;Elasticsearch的底层是由Lucene实…

鸿蒙主流路由详解

鸿蒙主流路由详解 Navigation Navigation更适合于一次开发,多端部署,也是官方主流推荐的一种路由控制方式,但是,使用起来入侵耦合度高,所以,一般会使用HMRouter,这也是官方主流推荐的路由 Navigation官网地址 个人源码地址 路由跳转 第一步-定义路由栈 Provide(PageInfo) pag…

java使用itext生成pdf

一、利用Adobe Acrobat DC软件创建pdf模板 备好Adobe Acrobat DC软件 1.excel/jpg/png文件转pdf文件 右击打开我们要转换的文件 2.然后点击 添加 域 3.可以看到域的名字 4.调整字体大小/对齐方式等 5.保存 二&#xff0c;代码部分 首先 上依赖 <dependency><group…

生成对抗网络模拟缺失数据,辅助PAMAP2数据集仿真实验

PAMAP2数据集是一个包含丰富身体活动信息的数据集&#xff0c;它为我们提供了一个理想的平台来开发和测试HAR模型。本文将从数据集的基本介绍开始&#xff0c;逐步引导大家通过数据分割、预处理、模型训练&#xff0c;到最终的性能评估&#xff0c;在接下来的章节中&#xff0c…

全面解析:HTML页面的加载全过程(一)--输入URL地址,与服务器建立连接

用户输入URL地址&#xff0c;与服务器建立连接 用户在浏览器地址栏输入一个URL 浏览器开始执行以下三步操作操作&#xff1a;url解析、DNS查询、TCP连接 第一步&#xff1a;URL解析 什么是URL&#xff1f; URL(Uniform Resource Locator&#xff0c;统一资源定位符)是互联网…

STM32总体架构简单介绍

目录 一、引言 二、STM32的总体架构 1、三个被动单元 &#xff08;1&#xff09;内部SRAM &#xff08;2&#xff09;内部闪存存储器 &#xff08;3&#xff09;AHB到APB的桥&#xff08;AHB to APBx&#xff09; 2、四个主动&#xff08;驱动&#xff09;单元 &#x…

postman 调用 下载接口(download)使用默认名称(response.txt 或随机名称)

官网地址&#xff1a;https://www.postman.com 介绍 Postman 是一款流行的 API 开发和测试工具&#xff0c;用于发送 HTTP 请求、测试接口、调试服务器响应以及进行 API 文档管理。它支持多种请求类型&#xff08;如 GET、POST、PUT、DELETE 等&#xff09;&#xff0c;并且功能…

JavaScript将至

JS是什么&#xff1f; 是一种运行在客户端&#xff08;浏览器&#xff09;的编程语言&#xff0c;实现人机交互效果 作用捏&#xff1f; 网页特效 (监听用户的一些行为让网页作出对应的反馈) 表单验证 (针对表单数据的合法性进行判断) 数据交互 (获取后台的数据, 渲染到前…

Vue.js 学习总结(13)—— Vue3 version 计数介绍

前言 Vue3.5 提出了两个重要概念&#xff1a;version计数和双向链表&#xff0c;作为在内存和计算方面性能提升的最大功臣。既然都重要&#xff0c;那就单挑 version 计数来介绍&#xff0c;它在依赖追踪过程中&#xff0c;起到快速判断依赖项有没有更新的作用&#xff0c;所以…

全面解析多种mfc140u.dll丢失的解决方法,五种方法详细解决

当你满心期待地打开某个常用软件&#xff0c;却突然弹出一个错误框&#xff0c;提示“mfc140u.dll丢失”&#xff0c;那一刻&#xff0c;你的好心情可能瞬间消失。这种情况在很多电脑用户的使用过程中都可能出现。无论是游戏玩家还是办公族&#xff0c;面对这个问题都可能不知所…

《Spring 实战:小型项目开发初体验》

一、引言 Spring 作为一款强大的 Java 开发框架&#xff0c;在小型项目开发中有着广泛的应用。本文将带你深入体验 Spring 在小型项目开发中的实战过程&#xff0c;从环境搭建到项目部署&#xff0c;全面展示 Spring 的魅力。 一、引言 Spring 作为一款强大的 Java 开发框架&…

Altium Designer学习笔记 11-15 原理图的封装 编译 检查 _PCB封装库的创建

基于Altium Designer 23学习版&#xff0c;四层板智能小车PCB 更多AD学习笔记&#xff1a;Altium Designer学习笔记 1-5 工程创建_元件库创建Altium Designer学习笔记 6-10 异性元件库创建_原理图绘制 目录 11、Value值的核对 12、封装的统一管理 13、原理图的编译设置和检查…

【Spring Boot】# 使用@Scheduled注解无法执行定时任务

1. 前言 在 Spring Boot中&#xff0c;使用Scheduled注解来定义定时任务时&#xff0c;定时任务不执行&#xff1b;或未在规定时间执行。 import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component;Component public c…

Spring Boot3远程调用工具RestClient

Spring Boot3.2之后web模块提供了一个新的远程调用工具RestClient&#xff0c;它的使用比RestTemplate方便&#xff0c;开箱即用&#xff0c;不需要单独注入到容器之中&#xff0c;友好的rest风格调用。下面简单的介绍一下该工具的使用。 一、写几个rest风格测试接口 RestCont…

VLAN是什么,一个好的网络为什么要划分VLAN呢?

前言 在上一篇中讲解了交换机的工作原理&#xff0c;知道了交换机处理数据的转发方式&#xff0c;其中有两种情况会以广播方式进行发送数据&#xff0c;第一种是目的MAC是全F的&#xff0c;以及组播MAC&#xff0c;第二种是未知单播帧&#xff0c;那这个会带来什么样的问题呢&…

flowable流程图详细绘制教程

文章目录 前言一、flowable是什么&#xff1f;回答下之前的问题 二、flowable-modeler使用1. 使用步骤2.开始绘制弄一个请假的流程 三 加载该流程总结 前言 flowable有些晦涩难懂的东西&#xff1a; 我最开始接触的时候,还是用的activity,当时觉得好复杂,那么这次经过我自己在…

蓝桥杯c++算法秒杀【6】之动态规划【上】(数字三角形、砝码称重(背包问题)、括号序列、组合数问题:::非常典型的必刷例题!!!)

下将以括号序列、组合数问题超级吧难的题为例子讲解动态规划 别忘了请点个赞收藏关注支持一下博主喵&#xff01;&#xff01;&#xff01;! ! ! ! &#xff01; 关注博主&#xff0c;更多蓝桥杯nice题目静待更新:) 动态规划 一、数字三角形 【问题描述】 上图给出了一…

[Python3学习笔记-基础语法] Python3 基础语法

本篇文章详细介绍Python3的基础语法&#xff0c;主要包括编码、标识符、Python保留字、注释、行缩进、多行语句、Number类型、字符串、空行、print打印等。 这些是Python最基础的东西&#xff0c;掌握好了才能更好的学习后续的内容。 有兴趣共同结伴学习Python的朋友&#xff0…

HDR视频技术之三:色度学与颜色空间

HDR 技术的第二个理论基础是色度学。从前面的内容中可以了解到&#xff0c;光学以及人类视觉感知模型为人类提供了解释与分析人类感知亮度的理论基础&#xff0c;但是 HDR 技术不仅仅关注于提升图像与视频的亮度范围&#xff0c;同时也关注于提供更加丰富的色彩。因此&#xff…

数据库MYSQL——表的设计

文章目录 前言三大范式&#xff1a;几种实体间的关系&#xff1a;一对一关系&#xff1a;一对多关系&#xff1a;多对多关系&#xff1a; 前言 之前的博客中我们讲解的是关于数据库的增删改查与约束的基本操作&#xff0c; 是在已经创建数据库&#xff0c;表之上的操作。 在实…