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

文章目录

  • 一、Qt创建线程的三种方法
  • 二、Qt并发:QtConcurrent介绍
  • 三、QtConcurrent run参数说明
  • 四、获取QtConcurrent的返回值
  • 五、C++其他线程技术介绍

一、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等。

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

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

相关文章

C/C++内存分布

1.内存分布简略图 2.全局变量和静态变量的区别 (1)局部静态变量&#xff1a;存储在数据段中&#xff0c;局部静态变量的作用域在当前函数中&#xff0c;出了函数就不能使用该变量&#xff0c;但局部静态变量的生命周期是在整个程序间&#xff0c;局部静态变量要运行到这一行才…

贝叶斯估计(1):期末大乱炖

写在前面&#xff01; 1 先验分布和后验分布 三种信息&#xff1a;总体信息、样本信息、先验信息 总体信息&#xff1a;“总体是正态分布”&#xff1b;样本信息&#xff1a;总体抽取的样本提供的信息&#xff0c;是最新鲜的信息&#xff1b;先验信息&#xff1a;在抽样之前就…

019-GeoGebra中级篇-GeoGebra的坐标系

GeoGebra作为一款强大的数学软件&#xff0c;支持多种坐标系的使用&#xff0c;包括但不限于&#xff1a;笛卡尔坐标系&#xff08;Cartesian Coordinate System&#xff09;、极坐标系&#xff08;Polar Coordinate System&#xff09;、参数坐标系&#xff08;Parametric Coo…

第二证券股市知识:股票填权是怎么回事?利好还是利空?

1、股票填权的含义 股票填权是指在除权除息之后的一段时刻内&#xff0c;假设多数投资者看好该个股&#xff0c;股票的价格超过除权除息的基准价就叫做填权。上市公司假设能持续分红&#xff0c;就会向市场传递积极信号&#xff0c;招引更多投资者买入&#xff0c;越来越多的投…

Thingsboard 系列之通过 ESP8266+MQTT 模拟设备上报数据到平台

前置工作 Thingsboard平台ESP 8266 NodeMCU 开发板IDE&#xff1a; Arduino 或 VScode 均可 服务端具体对接流程 系统管理员账号通过 Thingsboard 控制面板创建租户等信息并以租户账号登录 实体 —> 设备维护具体设备信息 创建完成后通过管理凭据修改或直接复制访问令牌…

磁致伸缩液位计的应用领域

磁致伸缩液位计作为一种高精度、高稳定性的液位测量设备&#xff0c;在众多行业中都有着广泛的应用。接下来&#xff0c;我们将从多个角度详细探讨磁致伸缩液位计在不同领域的应用情况。 石油化工行业 在石油化工行业中&#xff0c;磁致伸缩液位计主要用于储罐、反应器和管道等…

太实用了吧?手把手教你华为eNSP模拟器桥接真实网络!

号主&#xff1a;老杨丨11年资深网络工程师&#xff0c;更多网工提升干货&#xff0c;请关注公众号&#xff1a;网络工程师俱乐部 晚上好&#xff0c;我的网工朋友。 今天聊聊eNSP桥接正式网络&#xff0c;就是把eNSP桥接进真实的网络&#xff0c;利用我们的物理网卡通过实体路…

数学建模论文写作文档word

目录 1. 摘要写法1.1 确定题目与方法1.2 编写开头段落1.3 填写问题一1.4 重复步骤3填写其他问题1.5 编写结尾段落1.6 编写关键词 2. 问题重述2.1 问题背景2.2 问题提出 3. 问题分析4. 问题X模型的建立与求解5. 模型的分析5.1 灵敏度分析5.2 误差分析&#xff08;主要用于预测类…

linux基础—目录和文件操作

1&#xff0c;列出目录和文件的详细信息 ls: ls -l ls -lt 2&#xff0c;认识文件 第一列 左边的一组排序中&#xff0c;第一个字符是文件的类型&#xff0c;后面9个字符是文件的权限。 第一个字符主要有3种情况&#xff1a; d表示目录、-表示文件&#xff0c;l表示链接 第…

【回溯算法经典题目解析】

1. 什么是回溯算法 回溯算法是⼀种经典的递归算法&#xff0c;通常用于解决组合问题、排列问题和搜索问题等。 回溯算法的基本思想&#xff1a;从一个初始状态开始&#xff0c;按照⼀定的规则向前搜索&#xff0c;当搜索到某个状态⽆法前进时&#xff0c;回退到前⼀个状态&am…

背包问题转换

如何转换成背包问题呢&#xff0c;我们可以把每个质数当成一个重量 #define _CRT_SECURE_NO_WARNINGS #include<bits/stdc.h> using namespace std;#define int long long int record[1005]; void fun() {//record[2] 1;for (int i 2; i < 1000; i) {if (!record[…

JDBC和数据库连接池

1 JDBC概述 1.1 数据持久化 持久化(persistence)&#xff1a;把数据保存到可掉电式存储设备中以供之后使用。大多数情况下&#xff0c;特别是企业级应用&#xff0c;数据持久化意味着将内存中的数据保存到硬盘上加以”固化”&#xff0c;而持久化的实现过程大多通过各种关系数…

鸿蒙语言基础类库:【@ohos.url (URL字符串解析)】

URL字符串解析 说明&#xff1a; 本模块首批接口从API version 7开始支持。后续版本的新增接口&#xff0c;采用上角标单独标记接口的起始版本。开发前请熟悉鸿蒙开发指导文档&#xff1a;gitee.com/li-shizhen-skin/harmony-os/blob/master/README.md点击或者复制转到。 导入…

第一百四十九节 Java数据类型教程 - Java子字符串、字符串转换

Java数据类型教程 - Java子字符串 获取子字符串 我们可以使用substring()方法来获取字符串的子部分。 我们可以将开始索引作为参数&#xff0c;并返回一个从开始索引开始到字符串结尾的子串。 我们还可以将开始索引和结束索引作为参数。 它返回从开始索引开始的子字符串和小…

使用预加载库优化 PostgreSQL 函数#postgresql认证

在 POSTGRESQL 中执行函数和过程 为了理解 PostgreSQL 的工作原理&#xff0c;我们首先要看一个简单的函数调用。下一个清单显示了一些简单的PostGIS代码&#xff1a; PgSQL test# timing Timing is on. test# SELECT * FROM hans.points WHERE id 1;id │ …

【工具分享】零零信安攻击面管理平台

文章目录 00SEC-ASM™功能介绍功能演示 最近闲来无事&#xff0c;到处网上冲浪&#xff0c;无意间发现了长亭云图攻击面管理平台&#xff0c;无奈需要授权才能使用&#xff0c;于是就找到了平替&#xff1a;零零信安攻击面管理平台。 长亭云图攻击面管理平台&#xff1a;https:…

代码随想录-Day50

1143. 最长公共子序列 给定两个字符串 text1 和 text2&#xff0c;返回这两个字符串的最长 公共子序列 的长度。如果不存在 公共子序列 &#xff0c;返回 0 。 一个字符串的 子序列 是指这样一个新的字符串&#xff1a;它是由原字符串在不改变字符的相对顺序的情况下删除某些…

Kotlin linkedMapOf filterKeys

Kotlin linkedMapOf filterKeys fun main(args: Array<String>) {val lhm linkedMapOf<String, Any>(Pair("name", "phil"), //因为key相同都为 name&#xff0c;被后面的覆盖。Pair("year", 2024),Pair("name", "f…

【TB作品】51单片机 Proteus仿真 00013红外proteus仿真循迹避障小车

实验报告&#xff1a;智能小车系统设计与实现 一、背景介绍 本实验旨在设计并实现一个基于STC89C52单片机控制的智能小车系统。该系统通过超声波传感器进行避障&#xff0c;通过红外接收器实现远程控制&#xff0c;同时具备循迹功能。整个系统的核心是单片机&#xff0c;它通…

初识c++(命名空间,缺省参数,函数重载)

一、命名空间 1、namespace的意义 在C/C中&#xff0c;变量、函数和后面要学到的类都是大量存在的&#xff0c;这些变量、函数和类的名称将都存在于全 局作用域中&#xff0c;可能会导致很多冲突。使用命名空间的目的是对标识符的名称进行本地化&#xff0c;以避免命名 冲突…