【Qt 多线程+opencv 读取和显示图像】

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档


前言

提示:这里可以添加本文要记录的大概内容:

工程需要Qt多线程+opencv 结合信号与槽读取和显示图像

提示:以下是本篇文章正文内容,下面案例可供参考

一、例程

加入了信号与槽机制来处理多线程中的图像数据:

  1. 创建工作线程类:首先创建一个工作线程类,它从QThread继承并包含一个信号来发送图像数据。
class WorkerThread : public QThread
{
    Q_OBJECT

public:
    void run() override {
        // 读取图像数据
        cv::Mat image = cv::imread("path_to_image.jpg");
        emit imageReady(image); // 发出信号,传递图像数据
    }

signals:
    void imageReady(const cv::Mat &image); // 信号,传递图像数据
};
  1. 创建主窗口类:创建一个主窗口类,其中包含一个槽函数来接收工作线程发出的信号,并在主线程中显示图像。
class MainWindow : public QObject {
    Q_OBJECT

public:
    MainWindow() {
        // 创建工作线程实例并启动
        WorkerThread *thread = new WorkerThread();
        connect(thread, &WorkerThread::imageReady, this, &MainWindow::displayImage); // 连接信号和槽
        thread->start();
    }

public slots:
    void displayImage(const cv::Mat &image) {
        // 在主线程中显示图像
        cv::imshow("Image Display", image);
        cv::waitKey(0); // 等待用户按键,保持显示状态
    }
};
  1. 连接信号和槽:在主窗口类的构造函数中,我们使用connect函数将工作线程的imageReady信号连接到主窗口类的displayImage槽。这样,当工作线程完成图像读取并发出imageReady信号时,displayImage槽函数将被自动调用。
  2. 主函数:最后,在主函数中创建主窗口类的实例并运行应用程序。
int main(int argc, char *argv[]) {
    QCoreApplication a(argc, argv);
    MainWindow mainWindow;
    return a.exec();
}

这个例子展示了如何使用Qt的信号与槽机制在多线程环境中安全地处理图像数据。工作线程读取图像数据并通过信号发送,主线程通过槽函数接收并显示图像,确保了线程安全和正确的同步。

二、线程的开启和关闭

在Qt中,线程的开始和关闭可以通过以下方式实现:

  1. 启动线程

    • 定义工作类:通常,你会创建一个类来继承QThread,并重写其run()函数。在这个函数中,你可以定义线程应该执行的代码。
    • 实例化并启动线程:在主函数中,你可以实例化这个工作类,并调用其start()方法来启动线程。
  2. 关闭线程

    • 优雅地关闭线程:在线程的run()函数中,你可以使用一个标志变量(如volatile bool m_toStop)来判断是否需要从run()函数返回。在线程运行的过程中,你可以检查这个标志变量的值来决定是否需要停止线程。当线程需要停止时,设置这个标志变量为true。在线程的run()函数结束时,线程自然死亡。
    • 强制关闭线程:如果需要立即停止线程,可以使用thread->quit()thread->wait()方法。前者会尝试停止正在运行的线程,后者会等待线程结束。

请注意,关闭线程时需要确保线程已经完成了其任务,避免资源泄露或数据不一致的问题。同时,处理多线程时需要考虑到线程同步和互斥的问题,确保线程安全地访问共享资源。

以下是一个使用Qt多线程的详细例程:

#include <QCoreApplication>
#include <QThread>
#include <QDebug>
#include <QTimer>

class Worker : public QThread
{
    Q_OBJECT

public:
    Worker() {}
    void run() override {
        int i = 0;
        while (!isInterruptionRequested()) {
            qDebug() << "Thread is running" << ++i;
            QThread::sleep(1); // 模拟耗时操作
        }
    }
};

class MainWindow : public QObject {
    Q_OBJECT
public:
    MainWindow(QObject *parent = nullptr) : QObject(parent) {
        Worker *worker = new Worker();
        connect(worker, &Worker::started, this, &MainWindow::onWorkerStarted);
        connect(worker, &Worker::finished, this, &MainWindow::onWorkerFinished);
        worker->start(); // 启动线程
    }

public slots:
    void onWorkerStarted() { qDebug() << "Worker started"; }
    void onWorkerFinished() { qDebug() << "Worker finished"; }
};

int main(int argc, char *argv[]) {
    QCoreApplication a(argc, argv);
    MainWindow mainWindow;
    return a.exec();
}

这个例程中,我们创建了一个Worker类来继承QThread,并重写了其run()函数。在run()函数中,我们使用一个循环来模拟线程的持续运行,并每隔1秒打印一条消息。我们还定义了一个MainWindow类,其中包含一个工作线程的实例。通过连接工作线程的startedfinished信号到主窗口类的槽函数,我们可以在工作线程开始和结束时执行特定的操作。最后,在主函数中,我们创建了主窗口类的实例并运行应用程序。

三、判断线程是否还在运行

在Qt中,判断线程是否仍在运行可以通过检查线程的状态来实现。Qt提供了QThread::isRunning()函数,该函数返回一个布尔值,表示线程是否正在运行。

以下是一个简单的示例代码,演示如何使用isRunning()函数来判断线程是否仍在运行:

#include <QCoreApplication>
#include <QThread>
#include <QDebug>

class MyThread : public QThread {
    Q_OBJECT
public:
    MyThread(QObject *parent = nullptr) : QThread(parent) {}

    void run() override {
        // 线程执行的代码
        qDebug() << "Thread started";
        // ...
        // 线程结束前的操作
        qDebug() << "Thread finished";
        exit(0); // 结束线程
    }
};

int main(int argc, char *argv[]) {
    QCoreApplication a(argc, argv);
    MyThread thread;
    thread.start(); // 启动线程

    // 检查线程是否仍在运行
    while (thread.isRunning()) {
        qDebug() << "Thread is still running";
        QThread::sleep(1); // 等待一段时间,避免过度消耗CPU资源
    }
    qDebug() << "Thread has finished";

    return a.exec();
}

在上面的代码中,我们创建了一个自定义的MyThread类,它继承自QThread。在run()函数中,我们定义了线程要执行的代码。在主函数中,我们创建了MyThread的实例,并调用其start()方法来启动线程。然后,我们使用一个while循环来检查线程是否仍在运行。如果线程仍在运行,循环会继续执行并打印消息。在每次循环迭代之间,我们使用QThread::sleep(1)来等待一段时间,以避免过度消耗CPU资源。当线程结束时,isRunning()函数将返回false,循环将停止执行。最后,我们打印一条消息表示线程已经完成。

总结

  • 6
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您好!要在 Qt 中使用 OpenCV 线程读取摄像头视频,可以按照以下步骤操作: 1.首先,您需要在项目中包含 OpenCV 库。您可以通过在 .pro 文件中添加以下行来实现: ``` INCLUDEPATH += /usr/local/include/opencv4 LIBS += -L/usr/local/lib -lopencv_core -lopencv_highgui -lopencv_imgproc -lopencv_videoio ``` 注意:根据您的系统和 OpenCV 安装位置,您需要相应地更改路径。 2.创建一个新的类,例如 CameraThread,来处理摄像头视频的读取和处理。此类应该继承自 QThread。 3.在 CameraThread 类中,您可以使用 OpenCV 的 VideoCapture 类来读取摄像头视频。您可以按照以下示例代码编写 run() 函数: ``` void CameraThread::run() { cv::VideoCapture cap(0); // 0 表示摄像头设备编号,如果有多个摄像头,可以尝试其他编号 if (!cap.isOpened()) { qDebug() << "Error opening camera"; return; } cv::Mat frame; while (!isInterruptionRequested()) { cap.read(frame); // 读取一帧视频 if (frame.empty()) { qDebug() << "No frame captured"; break; } // 在这里处理视频帧,例如进行图像处理或显示 ... // 将 OpenCV Mat 转换为 Qt QImage 并发送到主线程 QImage img(frame.data, frame.cols, frame.rows, QImage::Format_RGB888); emit sendImage(img); } cap.release(); // 释放摄像头 } ``` 注意:在处理视频帧时,您可以使用 OpenCV图像处理函数来进行图像处理,例如 cv::cvtColor()、cv::GaussianBlur() 等等。 4.在 CameraThread 类中添加一个信号 sendImage(),用于将 Qt QImage 发送到主线程。例如: ``` signals: void sendImage(QImage); ``` 5.在主窗口中,您可以创建一个 QLabel 或 QGraphicsScene 来显示视频帧。然后,连接 CameraThread 的 sendImage() 信号到主窗口中的槽函数,例如: ``` connect(cameraThread, &CameraThread::sendImage, this, &MainWindow::updateImage); ``` 其中,updateImage() 函数用于更新 QLabel 或 QGraphicsScene 中的图像。 6.在主窗口中创建 CameraThread 对象,并启动线程。例如: ``` CameraThread *cameraThread = new CameraThread(); cameraThread->start(); ``` 注意:当您关闭应用程序时,应该停止线程并释放摄像头资源。例如,在主窗口的析构函数中添加以下代码: ``` cameraThread->requestInterruption(); cameraThread->wait(); delete cameraThread; ``` 希望这些步骤对您有所帮助!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值