首页 > 其他分享 >Qt开发 | Qt创建线程 | Qt并发-QtConcurrent

Qt开发 | Qt创建线程 | Qt并发-QtConcurrent

时间:2024-07-11 10:59:57浏览次数:22  
标签:run Qt int QtConcurrent concurrent 线程 include

文章目录

一、Qt创建线程的三种方法

  以下是Qt创建线程的三种方法:

  • 方法一:派生于QThread

    派生于QThread,这是Qt创建线程最常用的方法。线程类中重写虚函数void QThread::run();,在run()函数中写具体内容,外部通过start()调用,即可执行线程体run()

    注意:派生于QThread的类,构造函数属于主线程,run函数属于子线程,可以通过打印线程id来判断

    示例:

    VS中创建Qt控制台项目

    MyThread.h

    #pragma once
    
    #include <QThread>
    
    class MyThread : public QThread
    {
    public:
    	MyThread();
    	void run()override;
    };
    

    MyThread.cpp

    #include "MyThread.h"
    #include <QDebug>
    
    MyThread::MyThread()
    {
    	qDebug() << "MyThread construct id =" << QThread::currentThreadId;
    }
    
    void MyThread::run()
    {
    	qDebug() << "MyThread run id =" << QThread::currentThreadId;
    	int index = 0;
    	while (1)
    	{
    		qDebug() << index++;
    		QThread::msleep(500);
    	}
    }
    

    main.cpp

    #include <QtCore/QCoreApplication>
    #include <QDebug>
    #include "MyThread.h"
    
    
    int main(int argc, char *argv[])
    {
        QCoreApplication a(argc, argv);
        qDebug() << "main thread id = " << QThread::currentThreadId;
    
        MyThread th;
        th.start();
    
        qDebug() << "main thread end!";
        return a.exec();
    }
    

    运行结果

    main thread id =  0x78272815
    MyThread construct id = 0x78272815
    main thread end!
    MyThread run id = 0x78272815
    0
    1
    2
    3
    4
    5
    6
    ...
    
  • 方法二:派生于QRunnable

    派生于QRunnable,重写run()方法,在run()方法里处理其他任务,调用时需要借助Qt线程池

    注意:这种新建线程的方法的最大缺点是:不能使用Qt信号槽机制,因为QRunnable不是继承自QObject。但是这种方法的好处是:可以让QThreadPool来管理线程,QThreadPool会自动清理我们新建的QRunnable对象。

    示例:

    MyRunnable.h

    #pragma once
    
    #include <QRunnable>
    
    class MyRunnable : public QRunnable
    {
    public:
    	MyRunnable();
    	void run()override;
    };
    

    MyRunnable.cpp

    #include "MyRunnable.h"
    #include <QThread>
    #include <QDebug>
    
    MyRunnable::MyRunnable()
    {
    	qDebug() << "MyRunnable construct id =" << QThread::currentThreadId;
    }
    
    void MyRunnable::run()
    {
    	qDebug() << "MyRunnable run id =" << QThread::currentThreadId;
    }
    

    main.cpp

    #include <QtCore/QCoreApplication>
    #include <QDebug>
    #include <QThreadPool>
    #include "MyThread.h"
    #include "MyRunnable.h"
    
    
    int main(int argc, char *argv[])
    {
        QCoreApplication a(argc, argv);
        qDebug() << "main thread id = " << QThread::currentThreadId;
    
        //MyThread th;
        //th.start();
    
        MyRunnable* th = new MyRunnable();
        QThreadPool::globalInstance()->start(th);
    
        qDebug() << "main thread end!";
        return a.exec();
    }
    

    运行结果

    main thread id =  0x78272815
    MyRunnable construct id = 0x78272815
    main thread end!
    MyRunnable run id = 0x78272815
    
  • 方法三:moveToThread

    派生于QObject,使用moveToThread方法将任何QObject派生类的实例移动到另一个线程。将QThread对象作为私有成员,在构造函数里moveToThread,然后启动线程

    QThread *thread = new QThread;
    this->moveToThread(thread);
    thread->start();
    

    示例:

    MyObject.h

    #pragma once
    #include <qobject.h>
    #include <QThread>
    
    class MyObject : public QObject
    {
    	Q_OBJECT
    public:
    	MyObject();
    
    public slots:
    	void process();
    
    private:
    	QThread m_pTh;
    };
    

    MyObject.cpp

    #include "MyObject.h"
    #include <QDebug>
    
    MyObject::MyObject()
    {
    	this->moveToThread(&m_pTh);
    	m_pTh.start();
    
    	qDebug() << "MyObject construct id = " << QThread::currentThreadId();
    }
    
    void MyObject::process()
    {
    	qDebug() << "MyObject process id = " << QThread::currentThreadId();
    	
    	int index = 0;
    	while (1)
    	{
    		qDebug() << index++;
    		QThread::msleep(300);
    	}
    }
    

    ch7_1_mtt.h

    #pragma once
    
    #include <QtWidgets/QWidget>
    #include "ui_ch7_1_mtt.h"
    #include "MyObject.h"
    
    class ch7_1_mtt : public QWidget
    {
        Q_OBJECT
    
    public:
        ch7_1_mtt(QWidget *parent = nullptr);
        ~ch7_1_mtt();
    
    public slots:
        void on_pushButton_clicked();
    
    signals:
        void sig_pro();
    
    private:
        Ui::ch7_1_mttClass ui;
        MyObject* m_pObj;
    };
    

    ch7_1_mtt.cpp

    #include "ch7_1_mtt.h"
    #include <QDebug>
    
    
    ch7_1_mtt::ch7_1_mtt(QWidget *parent)
        : QWidget(parent)
    {
        ui.setupUi(this);
        qDebug() << "main construct id = " << QThread::currentThreadId();
    
        m_pObj = new MyObject();
    
        connect(this, &ch7_1_mtt::sig_pro, m_pObj, &MyObject::process);
    }
    
    ch7_1_mtt::~ch7_1_mtt()
    {}
    
    void ch7_1_mtt::on_pushButton_clicked()
    {
        //通过信号槽方式调用,相当于在子线程中运行;若直接调用process()函数,则相当于在主线程中运行
        emit sig_pro();
    }
    

    在槽函数中进行耗时操作可以使用线程,在子线程中进行耗时操作

    main.cpp

    #include "ch7_1_mtt.h"
    #include <QtWidgets/QApplication>
    
    int main(int argc, char *argv[])
    {
        QApplication a(argc, argv);
        ch7_1_mtt w;
        w.show();
        return a.exec();
    }
    

    运行结果

    main construct id = 0x52dc
    MyObject construct id =  0x52dc
    MyObject process id =  0x78272815
    0
    1
    2
    3
    4
    5
    6
    7
    

二、Qt并发:QtConcurrent介绍

  QtConcurrent是Qt框架中的一个模块,用于简化并发和多线程编程。它提供了一种高效的方式来执行并行任务,而无需直接处理线程的复杂性。QtConcurrent基于模板和函数式编程,使得编写并发代码更加简洁和易于理解。以下是QtConcurrent的一些关键特性:

  • 基于模板的并行算法:QtConcurrent提供了一系列的并行算法,如QtConcurrent::map()QtConcurrent::filter()QtConcurrent::reduce()等,它们可以并行地对数据集执行操作。
  • 基于Qt的线程池管理:QtConcurrent使用Qt的线程池来管理线程,这意味着开发者不需要手动创建和销毁线程,从而简化了线程管理。
  • 信号和槽的集成:QtConcurrent可以与Qt的信号和槽机制无缝集成,使得并行任务的结果可以很容易地通过信号传递给其他对象。
  • 懒加载和任务取消:QtConcurrent支持懒加载,即任务只有在实际需要结果时才开始执行。此外,它还支持取消正在执行的任务。
  • 异常处理:QtConcurrent可以处理并行执行过程中抛出的异常,确保程序的健壮性。
  • 无阻塞的等待:QtConcurrent提供了一种机制来等待并行任务的完成,而不会阻塞调用线程。
Header:
#include <QtConcurrent> 
qmake:
QT += concurrent

QtConcurrent基本用法:

  QtConcurrent::run这行代码的作用是启动ch72_concurrent类的timeTask成员函数在单独的线程中异步执行,并且返回一个QFuture<int>对象,用于跟踪执行的状态和获取返回值。

QFuture<int> ft = QtConcurrent::run(this, &ch72_concurrent::timeTask);

while (!ft.isFinished())
{
    //用于处理事件队列中的事件的。这个函数可以确保应用程序界面保持响应状态,即使在执行长时间运行的任务时
    //当future未完成时,让cpu去做别的事
    QApplication::processEvents(QEventLoop::AllEvents, 30); 
}

示例:

xx.h

#pragma once

#include <QtWidgets/QWidget>
#include "ui_ch72_concurrent.h"

class ch72_concurrent : public QWidget
{
    Q_OBJECT

public:
    ch72_concurrent(QWidget *parent = nullptr);
    ~ch72_concurrent();

    int timeTask();

private slots:
    void on_pushButton_clicked();

private:
    Ui::ch72_concurrentClass ui;
};

xx.cpp

#include "ch72_concurrent.h"
#include <QThread>
#include <QDebug>
#include <QtConcurrent>
#include <QFuture>

ch72_concurrent::ch72_concurrent(QWidget *parent)
    : QWidget(parent)
{
    ui.setupUi(this);
}

ch72_concurrent::~ch72_concurrent()
{}

int ch72_concurrent::timeTask()
{
    int num = 0;
    for (int i = 0; i < 1000000; i++)
    {
        num++;
        qDebug() << num;
    }

    return num;
}

void ch72_concurrent::on_pushButton_clicked()
{
    //timeTask();

    QFuture<int> ft = QtConcurrent::run(this, &ch72_concurrent::timeTask);

    while (!ft.isFinished())
    {
        //用于处理事件队列中的事件的。这个函数可以确保应用程序界面保持响应状态,即使在执行长时间运行的任务时
        QApplication::processEvents(QEventLoop::AllEvents, 30); 
    }
}

main.cpp

#include "ch72_concurrent.h"
#include <QtWidgets/QApplication>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    ch72_concurrent w;
    w.show();
    return a.exec();
}

三、QtConcurrent run参数说明

  QtConcurrent run函数参数,可以是全局函数,也可以是类成员函数。

  • 类成员函数做run参数

    int ch73_concurrent::timeTask(int num1, int num2)
    {
        //int num = 0;
        for (int i = 0; i < 1000000; i++)
        {
            num1++;
            num2++;
            qDebug() << num1;
            qDebug() << num2;
        }
    
        return num1 + num2;
    }
    
    void ch73_concurrent::on_pushButton_clicked()
    {
        //timeTask();
    
        int num1 = 0;
        int num2 = 0;
        QFuture<int> ft = QtConcurrent::run(this, &ch73_concurrent::timeTask, num1, num2);
    
        while (!ft.isFinished())
        {
            QApplication::processEvents(QEventLoop::AllEvents, 30);
        }
    }
    
  • 全局函数做run参数

    static int gTimeTask(int num1, int num2)
    {
        //int num = 0;
        for (int i = 0; i < 1000000; i++)
        {
            num1++;
            num2++;
            qDebug() << num1;
            qDebug() << num2;
        }
    
        return num1 + num2;
    }
    
    void ch73_concurrent::on_pushButton_clicked()
    {
        //timeTask();
    
        int num1 = 0;
        int num2 = 0;
        //QFuture<int> ft = QtConcurrent::run(this, &ch73_concurrent::timeTask, num1, num2);
        QFuture<int> ft = QtConcurrent::run(gTimeTask, num1, num2);
    
        while (!ft.isFinished())
        {
            QApplication::processEvents(QEventLoop::AllEvents, 30);
        }
    }
    

四、获取QtConcurrent的返回值

  获取QtConcurrent的结果,需要使用QFutureWatcher类,链接它的信号finished,然后给watcher设置future,当监控到future执行结束后,可以获取它的执行结果,调用的是result()函数。

int ch74::timeTask(int& num1, int& num2)
{
    for (int i = 0; i < 1000; i++)
    {
        num1++;
        num2++;
        qDebug() << num1;
        qDebug() << num2;
    }

    return num1 + num2;
}

void ch74::on_pushButton_clicked()
{
    int num1 = 0;
    int num2 = 0;

    QFutureWatcher<int>* fw = new QFutureWatcher<int>;

    connect(fw, &QFutureWatcher<int>::finished, [&]{
        qDebug() << "QFutureWatcher finished";
        qDebug() << "result = " << fw->result();
        });

    QFuture<int> ft = QtConcurrent::run(this, &ch74::timeTask, num1, num2);
    fw->setFuture(ft);

    while (!ft.isFinished())
    {
        QApplication::processEvents(QEventLoop::AllEvents, 30);
    }
}

五、C++其他线程技术介绍

  • pthread:linux线程
    • 这是POSIX线程库,主要用于Unix-like系统,如Linux和macOS。
    • 提供了丰富的线程创建和管理功能,包括互斥锁、条件变量等同步机制。
  • win32-pthread,obs的线程全部使用了win32-pthread
    • 这是一个在Windows平台上模拟POSIX线程库的库。
    • OBS(Open Broadcaster Software)等应用程序使用这个库来实现跨平台的线程功能。
  • windows thread类
    • Windows API提供了自己的线程类,如Win32 Thread,用于创建和管理线程。
    • 这些线程类通常与Windows特定的功能紧密结合,如窗口消息处理。
  • MFC thread类
    • MFC(Microsoft Foundation Classes)是一套C++库,用于Windows应用程序开发。
    • MFC提供了自己的线程类,例如CWinThread,用于简化线程的创建和管理。
  • boost
    • Boost库是一个广泛使用的C++库集合,其中包含了线程库。
    • Boost.Thread提供了跨平台的线程支持,包括线程创建、同步等。
  • std::thread(推荐使用这个,基于语言级别的跨平台C++线程)
    • C++11标准引入了std::thread,这是语言级别的线程支持。
    • 推荐使用std::thread,因为它提供了跨平台的线程支持,并且与C++标准库紧密集成。
    • std::thread简化了线程的创建和管理,同时提供了丰富的同步机制,如std::mutexstd::condition_variable等。

标签:run,Qt,int,QtConcurrent,concurrent,线程,include
From: https://blog.csdn.net/zwcslj/article/details/140278967

相关文章

  • QT窗口类
    1窗口基类我们在通过Qt向导窗口基于窗口的应用程序的项目过程中倒数第二步让我们选择跟随项目创建的第一个窗口的基类,下拉菜单中有三个选项,分别为:QMainWindow、QDialog、QWidget如下图:常用的窗口类有3个在创建Qt窗口的时候,需要让自己的窗口类继承上述三个窗口类的其......
  • Redis中设置增量缓存,减少对数据库的交互查询;启动@Async;异步线程
    //当属于这个分支的报文传入调用processMessage方法if((newJSONObject(dataMessage).optString("documentStatus")).equals("carWeizi_redis_service")){processMessage(dataMessage);}//processMessage中先把增量数据插入数据库,同时缓存redispublic......
  • lazarus 线程1
    标准线程用法TypeTAnalyzedataThread=class(TThread)privateprocedureDoProcedure;protectedprocedureExecute;override;publicConstructorCreate(CreateSuspended:boolean);end;constructorTAnalyzedataThread.Create(CreateSuspended:boolean);beginFree......
  • QT入门
    1QT概述1.1什么是QT是一个跨平台的C++应用程序开发框架具有短平快的优秀特质:投资少、周期短、见效快、效益高几乎支持所有的平台,可用于桌面程序开发以及嵌入式开发有属于自己的事件处理机制可以搞效率的开发基于窗口的应用程序。Qt是标准C++的扩展,C++的语法在Qt......
  • Qt入门(C++)
    创建项目基类的选择对于基类的选择有三个选项,分别是QMainWindow、QWidget、QDialog基类说明QMainWindow主窗⼝类,⼀般⽤于较为复杂的应⽤程序,除了中央客⼾区界⾯,还包括菜单栏、⼯具栏、状态栏以及多个可停靠的⼯具对话框等QWidget最简单、最基本的窗体程序,⾥⾯可以放置多......
  • Qt信号与槽
    信号和槽是QT自行定义的一种通信机制,独立于标准的C/C++语言。信号与槽可以一对一、一对多、多对一。 信号(signals:)是无返回值、无函数体、可有参的函数(声明),被emit发出后无序地被对应的槽接收然后执行槽函数。槽(slots:)是普通的类成员函数。 信号与槽的绑定函数原型:boolQOb......
  • JAVA高级之线程池
    线程的状态(6种)线程状态具体含义NEW一个尚未启动的线程的状态。也称之为初始状态、开始状态。线程刚被创建,但是并未启动。还没调用start方法。MyThreadt=newMyThread()只有线程象,没有线程特征。RUNNABLE当我们调用线程对象的start方法,那么此时线程对象进入了RUNNABLE状态。......
  • Redis是单线程还是多线程的?
    讲Redis是单线程还是多线程的需要根据redis各版本的一个变化,在Redis的老版本中,redis是单线程的,redis的数据处理读写命令都是由一个线程完成,并且速度很快,是因为redis的数据都是存储在内存中的,避免了磁盘I/O的瓶颈,有通过非阻塞IO和事件驱动模型,使得单线程依旧可以处理大量的数据......
  • Java Executors类的9种创建线程池的方法及应用场景分析
    在Java中,Executors类提供了多种静态工厂方法来创建不同类型的线程池。在学习线程池的过程中,一定避不开Executors类,掌握这个类的使用、原理、使用场景,对于实际项目开发时,运用自如,以下是一些常用的方法,来一一细说:newCachedThreadPool():创建一个可缓存的线程池,如果线程池中......
  • QtCreator使用Heob进行程序内存泄漏检测
    在Windows下,我们可以在QtCreator里使用Heob进行内存泄漏检测。下载到这里下载最新版的heob:【heob-heapobserver】下载后解压到某个文件夹下QtCreator使用heob新建一个Qt工程,然后写两句绝对导致内存泄漏的语句,并且要在profile或者debug模式下编译。然后选择Analyze--》......