Qt5教程(九):Qt多线程

article/2025/10/27 16:58:23

目录:

一、创建工程

二、QThread 源码一览

三、QThread相关方法介绍

四、创建线程

一、创建工程

先创建一个工程吧, 具体步骤前面讲过很多次了, 就不再细说了。

然后在Header文件夹下创建添加一个头文件, 右键Headers -> Add New... -> C++ -> C++ Header File -> Choose

随便起个名字, 比如mythread, 然后点Next->Finish。

本文福利, 免费领取Qt开发学习资料包、技术视频,内容包括(C++语言基础,Qt编程入门,QT图像,QT网络,QT数据库编程,QT项目实战,QT嵌入式开发,Quick模块等等)↓↓↓↓↓↓见下面↓↓文章底部点击免费领取↓↓

二. QThread 源码一览

在mythread.h中包含QThread头文件:

按住Ctrl键, 点击QThread, 再按住Ctrl键点击qthread.h进入到qthread.h文件, 源码就在这里了, 随便看看就好。哪里不懂就鼠标点一下不懂的地方, 然后按F1, 会跳转到相应的帮助文档,里面讲得很详细, 里面的英文也比较简单。

 

三、QThread相关方法介绍

3.1 启动线程

  • void start(Priority = InheritPriority);
    • 通过调用start()方法来启动线程,该方法会调用run()函数(可以看到QThread中run()为虚函数, 需要我们来重载)。

    • run()函数可调用exec()让该线程进入事件循环。

    • Priority为线程优先级(下面会讲)。

3.2 关闭线程

  • void exit(int retcode = 0);
    • 使线程退出事件循环, 如果该线程没有事件循环, 不做任何操作。
    • retcode默认为0, 表示正常返回。而非0值表示异常退出。
  • void quit();
    • 相当于exit(0)
  • void terminate();
    • 由操作系统强行终止该线程, 可能会导致无法完成一些清理工作, 不推荐使用。
  • void requestInterruption(); + bool isInterruptionRequested();
    • Qt5的新接口, requestInterruption用于请求线程进行中断。isInterruptionRequested返回true/false, 用于判断是否有终止线程的请求。

3.3 阻塞线程

  • bool wait(unsigned long time = ULONG_MAX);
    • 阻塞线程time毫秒, 默认永久阻塞;
    • 只有当线程结束(从run函数返回), 或阻塞超时才会返回;
    • 线程结束或还未启动, wait返回值为true, 超时的返回值为false。
  • static void sleep(unsigned long);
    • 阻塞xx秒, 无返回值。
  • static void msleep(unsigned long);
    • 阻塞xx毫秒, 无返回值。
  • static void usleep(unsigned long);
    • 阻塞xx微秒, 无返回值。

3.4线程状态判断

  • bool isFinished() const;
    • 如果线程结束返回true, 否则返回false。
  • bool isRunning() const;
    • 如果线程正在运行返回true, 否则返回false。
  • bool isInterruptionRequested() const;
    • 如果有终止线程的请求返回true, 否则返回false; 请求可由requestInterruption()发出。

3.5 设置优先级

  • void setPriority(Priority priority);

    • 用于设置正在运行的线程的优先级, 如果线程未运行, 则该返回不会执行任何操作并立刻返回。可用start(priority)启动带优先级的线程。

    • 指定的优先级是否生效取决于操作系统的调度, 如果是不支持线程优先级的系统上, 优先级的设置将被忽略。

    • 优先级可以设置为QThread::Priority内除InheritPriortyd的任何值:

      QThread::Priority枚举元素描述
      QThread::IdlePriority0没有其它线程运行时才调度
      QThread::LowestPriority1比LowPriority调度频率低
      QThread::LowPriority2比NormalPriority调度频率低
      QThread::NormalPriority3操作系统的默认优先级
      QThread::HighPriority4比NormalPriority调度频繁
      QThread::HighestPriority5比HighPriority调度频繁
      QThread::TimeCriticalPriority6尽可能频繁的调度
      QThread::InheritPriority7使用和创建线程同样的优先级(这是默认值)

3.6 信号

  • void started(QPrivateSignal);
    • 在线程start后, 执行run前发出该信号。
  • void finished(QPrivateSignal);
    • 在线程结束, 完全退出前发送此信号。

     

四、创建线程

4.1 继承QThread方式

a. 定义MyThread类

在mythread.h中定义MyThread类, 并继承QThread, 然后把框架写好:

#ifndef MYTHREAD_H
#define MYTHREAD_H#include <QThread>class MyThread : public QThread
{Q_OBJECTpublic:MyThread();private:protected:void run();signals:public slots:};#endif // MYTHREAD_H

b. 重载run()

新建一个C++ Source File, 命名为mythread.cpp

 mythread.cpp代码如下, run()函数中我们让它每隔1秒打印一次字符串:

#include "mythread.h"// 构造函数
MyThread::MyThread()
{}void MyThread::run()
{while (!isInterruptionRequested()){qDebug() << "Running...";sleep(1);}qDebug() << "Get Interruption Request, I'll exit.";
}

因为用到了qDebug(), 别忘了在mythread.h中添加<QDebug>头文件:

#include <QDebug>

c. 开始和结束线程

mainwindow.h中添加头文件和声明变量:

#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include "mythread.h"  // 添加头文件class MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = 0);~MainWindow();private:MyThread *my_thread;  // 声明变量
};#endif // MAINWINDOW_H

mainwindow.cpp中开启和结束线程:

#include "mainwindow.h"MainWindow::MainWindow(QWidget *parent): QMainWindow(parent)
{my_thread = new MyThread;  // 实例化my_thread->start();  // 开启线程// 主线程阻塞5秒QDateTime start = QDateTime::currentDateTime();QDateTime now;do {now = QDateTime::currentDateTime();} while(start.secsTo(now) < 5);// 关闭线程my_thread->requestInterruption();my_thread->wait();}MainWindow::~MainWindow()
{}

因为用到了<QDateTime>, 别忘了在mainwindow.h中添加头文件:

#include <QDateTime>

运行结果:

可以看到主线程被阻塞了5秒, 之后才弹出窗口。但是在主线程阻塞期间, 我们的my_thread线程仍在运行, 直到线程被关闭。

附: Qt4适用写法

上面我们结束线程使用的是requestInterruption()isInterruptionRequested(), 这是Qt5新增的, 那么Qt4要如何结束线程呢?

  • 首先需要使用一个flag来标识线程的状态(执行还是停止), 比如定义一个变量bool is_stopped 初值赋为false;
  • 然后自己写一个结束线程的函数, 比如stop(), 当调用my_thread->stop();时将is_stopped改为true;
  • run()中判断, 如果is_stoppedfalse线程继续执行, 如果为true线程退出; 别忘了退出前再将is_stopped改为false, 不然线程没法再次开启了。

代码如下:

mythread.h

#ifndef MYTHREAD_H
#define MYTHREAD_H#include <QThread>
#include <QDebug>class MyThread : public QThread
{Q_OBJECTpublic:MyThread();void stop();  // 添加stop()方法private:volatile bool is_stopped;  // 添加标识变量protected:void run();signals:public slots:};#endif // MYTHREAD_H

mythread.cpp

#include "mythread.h"// 构造函数
MyThread::MyThread()
{is_stopped = false;  // 初始化标识变量
}void MyThread::run()
{while (!is_stopped)  // 更改判断条件{qDebug() << "Running...";sleep(1);}qDebug() << "is_stopped is true, I'll exit.";is_stopped = false;  // 重置变量值
}// 关闭线程
void MyThread::stop()
{is_stopped = true;
}

mainwindow.cpp

#include "mainwindow.h"MainWindow::MainWindow(QWidget *parent): QMainWindow(parent)
{my_thread = new MyThread;  // 实例化my_thread->start();  // 开启线程// 主线程阻塞5秒QDateTime start = QDateTime::currentDateTime();QDateTime now;do {now = QDateTime::currentDateTime();} while(start.secsTo(now) < 5);// 关闭线程my_thread->stop();  // 用自己定义的方法关闭线程my_thread->wait();}MainWindow::~MainWindow()
{}

附: exit()和requestInterruption()区别

看例子, 我们修改一下run()函数和关闭线程部分的代码:

mythread.cpp

void MyThread::run()
{while (!isInterruptionRequested()){qDebug() << "Running...";sleep(1);}qDebug() << "子线程: 我只退出了while循环, 没有真正结束";exec();  // 事件循环qDebug() << "子线程: 我真的要结束了";
}

mainwindow.cpp

	// 关闭线程my_thread->requestInterruption();qDebug() << "主线程: 发起中断请求";my_thread->wait(3000);my_thread->quit();qDebug() << "主线程: 请求退出线程的事件循环";my_thread->wait();  // 等待线程结束

运行结果:

在主进程requestInterruption()后, 只是使得isInterruptionRequested()变为true, 退出了while循环, 在主线程中调用wait(3000), 并没有立刻返回, 而是3秒超时后才返回, 说明子线程没有真正结束, 而是执行到了exec()处进行事件循环。通过调用quit()exit()来结束子线程的事件循环, 子线程才真的结束了。

3.2 moveToThread方式(Qt5新增 官方推荐)

a. 定义一个继承QObject的类

  • 首先, 创建一个类并继承QObject, 把要在线程中执行的工作作为类的槽函数:

dowork.h

#ifndef DOWORK_H
#define DOWORK_H#include <QObject>
#include <QDateTime>
#include <QDebug>class DoWork : public QObject
{Q_OBJECT
public:explicit DoWork(QObject *parent = nullptr);public slots:void do_something();
};#endif // DOWORK_H

dowork.cpp

#include "dowork.h"DoWork::DoWork(QObject *parent) : QObject(parent)
{}void DoWork::do_something()
{int a = 5;while(a--){qDebug() << "Doing something ...";QDateTime start = QDateTime::currentDateTime();QDateTime now;do {now = QDateTime::currentDateTime();} while(start.secsTo(now) < 1);}qDebug() << "Done";
}

b. moveToThread

  • 然后, 创建一个线程对象, 把work1对象移到新线程下:

mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QThread>
#include "dowork.h"class MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = 0);~MainWindow();private:DoWork *work1;  // 自定义的类QThread *new_thread;  // 新线程
};#endif // MAINWINDOW_H

mainwindow.cpp

MainWindow::MainWindow(QWidget *parent): QMainWindow(parent)
{// 实例化work1 = new DoWork;new_thread = new QThread;work1->moveToThread(new_thread);  // 搬到线程下
}

c. 启动线程

  • 绑定线程启动后要做的工作:

    connect(new_thread, &QThread::started, work1, &DoWork::do_something);
    

    使用moveToThread的方法非常灵活, 你不一定要用&QThread::started来触发do_something, 也可以使用自定义的信号, 为了例程简单明了, 这里不举例了。

  • 启动线程

    new_thread->start();
    

d. 结束后的清理工作

  • 为了更安全, 线程结束后别忘了释放资源:

    connect(new_thread, &QThread::finished, work1, &QObject::deleteLater);
    
    MainWindow::~MainWindow()
    {new_thread->requestInterruption();new_thread->quit();new_thread->wait();
    }
    

附: mainwindow.cpp 完整代码

mainwindow.cpp

#include "mainwindow.h"MainWindow::MainWindow(QWidget *parent): QMainWindow(parent)
{// 实例化work1 = new DoWork;new_thread = new QThread;work1->moveToThread(new_thread);  // 搬到线程下connect(new_thread, &QThread::started, work1, &DoWork::do_something);connect(new_thread, &QThread::finished, work1, &QObject::deleteLater);new_thread->start();
}MainWindow::~MainWindow()
{new_thread->requestInterruption();new_thread->quit();new_thread->wait();
}

运行结果如下:

 本文福利, 免费领取Qt开发学习资料包、技术视频,内容包括(C++语言基础,Qt编程入门,QT图像,QT网络,QT数据库编程,QT项目实战,QT嵌入式开发,Quick模块等等)↓↓↓↓↓↓见下面↓↓文章底部点击免费领取↓↓


http://chatgpt.dhexx.cn/article/a8wU4xbd.shtml

相关文章

Qt 多线程的几种实现方式

Qt多线程的实现方式有&#xff1a; 1. 继承QThread类&#xff0c;重写run()方法 2. 使用moveToThread将一个继承QObject的子类移至线程&#xff0c;内部槽函数均在线程中执行 3. 使用QThreadPool,搭配QRunnable&#xff08;线程池&#xff09; 4. 使用QtConcurrent&#xf…

QT中的多线程

目录 1、QThread介绍 1.2、继承Qthread类 1.2.1、得到线程id 1.2.2、让线程一直执行 1.2.3、线程退出 1.2.4、​​​​​​​connect的第五个参数 1.2.5、线程锁 QMutex ​​​​​​​ 1、QThread介绍 QThread类提供了一个与平台无关的管理线程的方法。一个QThread对…

faild to create process解决办法

win7下运行pip时报faild to create process的解决办法&#xff1a; 1.找到python.exe文件&#xff0c;复制其文件路径。如下图&#xff0c;python.exe路径为D:\interpreter\python.exe。 2.找到pip-script.py文件&#xff0c;打开。将步骤1中复制的路径粘贴在第一行&#xff0…

CreateProcess error=2, 系统找不到指定的文件 解决方法

CreateProcess error2, 系统找不到指定的文件 解决方法 一般这是由于ndk缺少文件引起的 解决方法1&#xff1a; 在项目根目录下的local.properties文件中加cmd后缀 方法二下载16b的版本替换原来的ndk-bundle目录&#xff0c;默认在C:\Users\用户名\AppData\Local\Android\S…

SQL SERVER 2008 执行xp_cmdshell的过程中出错,调用createprocess失败,错误代码 5 解决方案

1、进入 控制面板----管理工具------本地安全策略 点击 本地策略--------安全选项 把“网络安全:LAN管理器身份验证级别”双击打开&#xff0c;选择“发送 LM 和 NTLM 响应”&#xff08;原来是“没有定义”&#xff09;。确认后&#xff0c;重新启动服务器&#xff0c;该问题…

runnerw.exe: CreateProcess failed with error 5:

在idea配置Git时遇到这个问题&#xff1a; 后来发现是我路径设置错了&#xff1a; 解决方案 修改设置里的路径即可&#xff08;Setings-Version Control-Git&#xff09;

Qt creater出现“启动程序失败,路径或者权限错误”或“The process could not be started!”解决方法

Qt creater出现“启动程序失败&#xff0c;路径或者权限错误”或“The process could not be started&#xff01;”解决方法 错误示例 远古版本的QTCreator在编译程序时没问题&#xff0c;在运行程序时会提示“The process could not be started&#xff01;" 较新版本…

failed to create process.

由于python-2.7是先出来&#xff0c;但是官方只更新到2020年1月1日&#xff1b;而python-3.6与python-2.7有一定的差别&#xff0c;会同时在电脑上安装这两个版本的python&#xff0c;为了能更好的调用python不同的版本&#xff0c;会设定一个执行时使用python2.exe&#xff0c…

CreateProcessAsUser

该CreateProcessAsUser函数创建一个新的进程及其主线程。新进程然后执行指定的可执行文件。该CreateProcessAsUser功能类似的CreateProcess函数&#xff0c;除了新进程运行在由hToken参数表示的用户的安全上下文中。默认情况下&#xff0c;新进程是非交互式的&#xff0c;即它运…

生成了文件却还是报错 Error:CreateProcess failed

想起来&#xff0c;以前用keil编译器的时候&#xff0c;也出现过这样的问题&#xff1a; 第一眼都是看到了“1 Error(s)”&#xff0c;就下意识认为自己程序出错了&#xff0c;找了半天没找到。 后来多看了一眼&#xff0c;发现完全这个错误其实可以完全不用理会&#xff0c;因…

【Error】Error running process: CreateProcess failed. Code 2

重新安装了pycham发现在pycharm里打不开控制台窗口了 解决办法&#xff1a; 在file -->setting --> Tools --> Terminal里 把Shell path 从 powershell.exe 改为 cmd.exe

Cannot run program “svn“ (in directory “D:xxxx“): CreateProcess error=2, 系统找不到指定的文件。报错解决方法

目录 一、报错内容 二 、原因 三、解决方法 一、报错内容 Cannot run program "svn" (in directory "xxxx"): CreateProcess error2, 系统找不到指定的文件 二 、原因 TortoiseSVN安装的时候没有安装command命令&#xff0c;不能通过命令执行版本管理…

CreateProcess error=5, 拒绝访问

一、情景再现 在一个Java项目中引入一个Python执行脚本&#xff0c;运行项目时报错&#xff1a; 二、问题分析 查看代码调用了源码中的exec&#xff08;&#xff09;方法 在java中&#xff0c;RunTime.getRuntime().exec()实现了调用服务器命令脚本来执行功能需要。也就是说…

failed to create process 的解决方式

网上大多数都是说python.exe移动了位置&#xff0c;导致你本来环境变量中配置的scripts中的python.exe找不到位置。 你需要修改scripts中的pip.exe中的python路径 如 当然这是其中一种可能性&#xff0c; 也有可能是由于你自己配置的环境变量产生了冲突 在windows的系统环境…

CreateProcess 创建进程失败原因调查

使用 CreateProcess 函数创建进程&#xff08;调用外部程序&#xff09;算是很常用的操作了&#xff0c;最近在工作中却遇到一个少见的怪现象&#xff0c;经常使用的一段代码&#xff08;调用外部程序并等待其结束&#xff0c;主要就使用了 CreateProcess 函数&#xff09;&…

jsp页面报org.apache.jasper.JasperException错误的一种解决方法

最近idea搭建ssm项目jsp页面出现如下问题 解决办法如下&#xff1a; 把 <% taglib uri“http://java.sun.com/jstl/core” prefix“c” %> 改成 <% taglib uri“http://java.sun.com/jstl/core_rt” prefix“c” %>

报org.apache.jasper.JasperException错误的一种解决方法

最近遇到的一个问题 把 <% taglib uri"http://java.sun.com/jstl/core" prefix"c" %> 改成 <% taglib uri"http://java.sun.com/jstl/core_rt" prefix"c" %> 这只是一种方法&#xff0c;如果解决不了&#xff0c;我…

org.apache.jasper.JasperException: java.lang.ClassNotFoundException解决办法

一、运行项目出错截图 报错找不到jsp文件&#xff0c;此时修改一下访问路径&#xff0c;会报如下错误: 二、解决java.lang.NoClassDefFoundError: javax/servlet/jsp/jstl/core/ConditionalTagSupport的方法&#xff1a; 1.缺少了相应的jar包&#xff1a; jstl-1.2standard…

org.apache.jasper.JasperException:解决办法!

org.apache.jasper.JasperException: The absolute uri: http://itcast.cn/common/ cannot be resolved in either web.xml or the jar files deployed with this application解决办法 今天在完成SSM框架整合的时候&#xff0c;使用eclipse在tomcat运行项目没提示任何问题&…

org.apache.jasper.JasperException: 无法在web.xml或使用此应用程序部署的jar文件中解析绝对uri:[http://java.sun.com/jsp/jstl/

问题如下&#xff1a; lib包下配置 jsp页面的导入一切正常&#xff0c;没有提示任何错误 网上也查了好多结果&#xff0c;最后发现修改tomcat中的conf/catalina.properties文件&#xff0c; tomcat.util.scan.DefaultJarScanner.jarsToSkip*.jar 改为&#xff1a; tomcat.ut…