Qt线程:QThread

article/2025/10/16 8:45:35

一、描述

一个QThread对象管理程序内的一个线程,QThreads在run()中开始执行。默认情况下,run()通过调用exec()启动事件循环,并在线程内部运行一个Qt事件循环。

可以通过使用 QObject::moveToThread() 将对象移动到线程来使用它们。

 class Worker : public QObject{Q_OBJECTpublic slots:void doWork(const QString &parameter){QString result;/* ... 这里是昂贵的或阻塞的操作 ... * /emit resultReady(result);}signals:void resultReady(const QString &result);};class Controller : public QObject{Q_OBJECTQThread workerThread;public:Controller() {Worker *worker = new Worker;worker->moveToThread(&workerThread);connect(&workerThread, &QThread::finished, worker, &QObject::deleteLater);connect(this, &Controller::operate, worker, &Worker::doWork);connect(worker, &Worker::resultReady, this, &Controller::handleResults);workerThread.start();}~Controller(){workerThread.quit();workerThread.wait();}public slots:void handleResults(const QString &);signals:void operate(const QString &);};

Worker 槽内的代码将在单独的线程中执行。可以自由地将 Worker 槽连接到来自任何对象、任何线程中的任何信号。跨不同线程连接信号和槽是安全的。

另一种使代码在单独线程中运行的方法是将 QThread 子类化并重新实现 run()。例如:

 class WorkerThread : public QThread{Q_OBJECTvoid run() override {QString result;/* ... 这里是昂贵的或阻塞的操作 ... * /emit resultReady(result);}signals:void resultReady(const QString &s);};void MyObject::startWorkInAThread(){WorkerThread *workerThread = new WorkerThread(this);connect(workerThread, &WorkerThread::resultReady, this, &MyObject::handleResults);connect(workerThread, &WorkerThread::finished, workerThread, &QObject::deleteLater);workerThread->start();}

示例中,线程将在 run 函数返回后退出。除非调用 exec(),否则线程中不会运行任何事件循环。

QThread 实例除了run()函数在新线程中执行,其他所有函数都在QThread 实例所在的线程中执行。这样可能会则从两个不同的线程访问同一个成员变量变量,要注意检查这样做是否安全。


二、类型成员

1、enum QThread::Priority:此枚举类型指示操作系统应如何调度新创建的线程。

  • IdlePriority:仅当没有其他线程正在运行时才调度。
  • LowestPriority:调度频率低于 LowPriority。
  • LowPriority:调度频率低于 NormalPriority。
  • NormalPriority:操作系统的默认优先级。
  • HighPriority:比 NormalPriority 更频繁地调度。
  • HighestPriority:比 HighPriority 更频繁地调度。
  • TimeCriticalPriority:尽可能经常安排调度。
  • InheritPriority:使用与创建线程相同的优先级。这是默认设置。

三、成员函数

1、[signal] void finished()

该信号在线程完成执行之前发出。当这个信号发出时,事件循环已经停止运行。除了延迟删除事件外,线程中将不再处理更多事件。该信号可以连接到 QObject::deleteLater(),以释放该线程中的对象。

注意:这是私有信号。 它可以用于信号连接,但不能由用户发射。

 2、void quit()

告诉线程的事件循环以返回码 0(成功)退出。 相当于调用 exit(0)。如果线程没有事件循环,则此函数不执行任何操作。此函数是线程安全的。

3、void start(QThread::Priority priority = InheritPriority)

通过调用 run() 开始执行线程。 操作系统会根据参数中设置的优先级调度线程。如果线程已经在运行,这个函数什么都不做。

优先级参数的效果取决于操作系统的调度策略。在不支持线程优先级的系统上,优先级将被忽略(如 Linux系统 )。

4、[signalvoid started()

当相关线程开始执行时,在调用 run() 函数之前,该信号从相关线程发出。这是私有信号,可以用于信号连接,但不能由用户发射。

5、void terminate()

终止线程的执行。线程可能会也可能不会立即终止,这取决于操作系统的调度策略。确保在terminate()之后调用 wait()。

当线程终止时,所有等待线程完成的线程都会被唤醒。

警告:此功能很危险,不鼓励使用。线程可能在其代码路径中的任何一点终止。修改数据时可能终止线程,线程将没有机会自行清理,解锁任何持有的互斥锁等。简而言之,只有在绝对必要时才使用此功能。

可以通过调用 setTerminationEnabled() 显式启用或禁用终止。在禁用终止时调用此函数会导致终止被推迟,直到重新启用终止。

6、~QThread()

销毁线程。删除 QThread 对象不会停止它管理的线程的执行。删除正在运行的 QThread将导致程序崩溃。应在删除 QThread 之前等待finished()信号。

7、[static] QThread *create(Function &&f, Args &&... args)

      [static] QThread *create(Function &&f)

创建一个新的 QThread 对象,该对象将使用参数 args 执行函数 f。

新线程没有启动,它必须通过显式调用 start() 启动。

注意:此函数仅在使用 C++17 时可用。

警告:不要在返回的 QThread 实例上多次调用 start(), 这样做会导致未定义的行为。

例:

int main(int argc, char *argv[])
{QApplication a(argc, argv);int i = 0;QThread * t = QThread::create([i]()mutable{while(i < 3){qDebug()<<i;++i;}qDebug()<<"lambda创建的子线程ID:"<<QThread::currentThreadId();});QObject::connect(t,&QThread::finished,[t]{qDebug()<<"线程是否仍在运行:"<<t->isRunning();t->deleteLater();});qDebug()<<"主线程ID:"<<QThread::currentThreadId();t->start();
}

 8、[static] QThread *currentThread()

返回指向当前执行线程的指针。

9、[static] Qt::HANDLE currentThreadId() 

返回当前正在执行的线程的线程句柄。警告:此函数返回的句柄用于内部目的,不应在任何应用程序代码中使用。

10、int exec()

进入事件循环并等待 exit() 被调用,返回传递给 exit() 的值。如果 exit() 是通过 quit() 调用的,则返回的值为 0。此函数从 run() 内部调用。需要调用这个函数来启动事件处理。

11、void exit(int returnCode = 0)

线程的事件循环以返回码returnCode 退出。

调用此函数后,线程离开事件循环并从对 QEventLoop::exec() 的调用返回。QEventLoop::exec() 函数返回 returnCode。

按照惯例,returnCode 为 0 表示成功,任何非零值表示错误。

12、[static] int idealThreadCount()

返回可以在系统上运行的理想线程数。这是通过查询系统中的实际和逻辑处理器内核数量来完成的。 如果无法检测到处理器内核数,则此函数返回 1。

13、bool isInterruptionRequested()

如果应停止在此线程上运行的任务,则返回 true。 requestInterruption() 可以请求中断。

此函数可用于使长时间运行的任务完全可中断。注意不要太频繁地调用它,以保持低开销。只能在线程本身内调用。

在长时间执行的任务中加上这个判断。如果有很消耗资源的情况可以中断线程。

14、void requestInterruption()

请求中断线程。该请求是建议性的,由线程上运行的代码决定是否以及如何应对此类请求。此函数不会停止在线程上运行的事件循环。

15、int loopLevel()

返回线程的当前事件循环级别。只能在线程本身内调用,即当它是当前线程时。

16、[static] void msleep(unsigned long msecs) (毫秒)

        [static] void sleep(unsigned long secs) (秒)

        [static] void usleep(unsigned long usecs) (微秒)

强制当前线程休眠。如果需要等待某个条件的达成,则应避免使用此函数改用信号槽。此功能不保证准确性。

17、void run()

线程的起点。调用 start() 后,新创建的线程调用此函数。默认实现只是调用 exec()。可以重新实现此功能以自定义线程操作。从此方法返回将结束线程的执行。

18、void setEventDispatcher(QAbstractEventDispatcher *eventDispatcher)

设置线程的事件调度程序。在使用 start() 启动线程之前,或者在主线程的情况下,在实例化 QCoreApplication 之前调用才有效。

19、void setStackSize(uint stackSize)

将线程的最大堆栈大小设置为 stackSize。 如果 stackSize 大于零,则最大堆栈大小设置为 stackSize 字节,否则最大堆栈大小由操作系统自动确定。

警告:大多数操作系统对线程堆栈大小设置了最小和最大限制。 如果堆栈大小超出这些限制,线程将无法启动。

20、[static protected] void setTerminationEnabled(bool enabled = true)

启用或禁用当前线程的终止该线程必须已由 QThread 启动。

当 enabled 为 false 时,终止被禁用。以后对 terminate() 的调用将不会生效。直到设置setTerminationEnabled(true)时才会生效
当 enabled 为 true 时,终止被启用。以后对 terminate() 的调用将正常终止线程。

21、bool wait(QDeadlineTimer deadline = QDeadlineTimer(QDeadlineTimer::Forever))

        bool wait(unsigned long time)

阻塞当前调用QThread对象的线程,直到满足以下任一条件:

  • QThread对象关联的线程已完成执行(即当它从 run() 返回时)。如果线程已完成执行,此函数将返回 true。如果线程尚未启动,它也会返回 true。
  • 截止时间已到。如果达到最后期限,此函数将返回 false(线程还在继续执行)。

设置为 QDeadlineTimer::Forever(默认值)的截止时间计时器永远不会超时,在这种情况下,该函数仅在线程从 run() 返回或线程尚未启动时返回。

class myThread : public QThread
{
public:myThread(){}
protected:void run()override{int n = 0;for(int i = 0;i < 2000000000;++i){if(i % 266 == 0)++n;}qDebug()<<"n = "<<n;}
};int main(int argc, char *argv[])
{QApplication a(argc,argv);myThread thread;thread.start();qDebug()<<thread.wait(1000);qDebug()<<"runing";Widget w;w.show();a.exec();
}

 22、[static] void yieldCurrentThread()

将当前线程的执行交给另一个可运行线程(如果有)。操作系统决定切换到哪个线程。


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

相关文章

PyQt中的多线程QThread示例

PyQt中的多线程 一、PyQt中的多线程二、创建线程2.1 设计ui界面2.2 设计工作线程2.3 主程序设计 三、运行结果示例 一、PyQt中的多线程 传统的图形用户界面应用程序都只有一个执行线程&#xff0c;并且一次只执行一个操作。如果用户从用户界面中调用一个比较耗时的操作&#x…

【Qt】Qt的线程(两种QThread类的详细使用方式)

Qt提供QThread类以进行多任务处理。与多任务处理一样&#xff0c;Qt提供的线程可以做到单个线程做不到的事情。例如&#xff0c;网络应用程序中&#xff0c;可以使用线程处理多种连接器。 QThread继承自QObject类&#xff0c;且提供QMutex类以实现同步。线程和进程共享全局变量…

Qt - 一文理解QThread多线程(万字剖析整理)

目录 为什么需要多线程QThread使用方法new QThread Class & Override run()new Object Class & moveToThread(new QThread) connect事件循环源码分析如何正确退出线程堆栈大小优先级线程间通讯线程同步互斥锁读写锁信号量条件变量 可重入与线程安全QObject的可重入性开…

Qt 线程中QThread的使用

文章目录 Qt 线程中QThread的使用1. 线程类 QThread1.1 常用共用成员函数1.2 信号槽1.3 静态函数1.4 任务处理函数 2. 使用方式 12.2 示例代码3. 使用方式 23.1 操作步骤3.2 示例代码 Qt 线程中QThread的使用 在进行桌面应用程序开发的时候&#xff0c; 假设应用程序在某些情况…

Qt之QThread(深入理解)

简述 为了让程序尽快响应用户操作&#xff0c;在开发应用程序时经常会使用到线程。对于耗时操作如果不使用线程&#xff0c;UI界面将会长时间处于停滞状态&#xff0c;这种情况是用户非常不愿意看到的&#xff0c;我们可以用线程来解决这个问题。 前面&#xff0c;已经介绍了…

Qt之QThread介绍(常用接口及实现、自动释放内存、关闭窗口时停止线程运行、同步互斥)

在程序设计中&#xff0c;为了不影响主程序的执行&#xff0c;常常把耗时操作放到一个单独的线程中执行。Qt对多线程操作有着完整的支持&#xff0c;Qt中通过继承QThread并重写run()方法的方式实现多线程代码的编写。针对线程之间的同步与互斥问题&#xff0c;Qt还提供了QMutex…

Qt线程QThread开启和安全退出

1、线程开启 Qt中&#xff0c;开启子线程&#xff0c;一般有两种方法&#xff1a; a, 定义工作类worker: worker继承 QThread, 重写run函数&#xff0c;在主线程中实例化worker&#xff0c;把耗时工作放进worker的run函数中完成&#xff0c;结束后&#xff0c;往主线程中发信…

QThread的用法

概述 QThread类提供了一个与平台无关的管理线程的方法。一个QThread对象管理一个线程。QThread的执行从run()函数的执行开始&#xff0c;在Qt自带的QThread类中&#xff0c;run()函数通过调用exec()函数来启动事件循环机制&#xff0c;并且在线程内部处理Qt的事件。在Qt中建立线…

Oracle 定时任务执行存储过程【建议收藏】

首先用一个完整的例子来实现定时执行存储过程。 任务目标&#xff1a;每小时向test表中插入一条数据。 实现方案&#xff1a; 1.通过 oracle 中 dbms_job 完成存储过程的定时调用 2.在存储过程中完成相应的逻辑操作 实现步骤&#xff1a; 1.创建一个测试表 create table test…

【Mysql】MySQL 用户执行存储过程的权限

问题 运行存储过程报错&#xff1a; 原因 查询资料&#xff1a; 1305错误&#xff0c;由于当前用户没用权限&#xff0c;对用户进行授权后可以执行。 解决 MySQL创建存储过程/函数需要的权限&#xff1a; alter routine---修改与删除存储过程/函数 create routine--创建…

goland 使用 gorm 执行 存储过程 : go语言 执行存储过程

使用 gorm 执行 存储过程 初安装依赖代码&#xff1a; 附存储过程图片存储过程代码&#xff08;创建&#xff09; 表结构表结构图表结构代码 初 最近遇到要写存储过程需求&#xff0c;使用 大佬写的 框架 gorm 来完成。简直是方便的不行&#xff1a; 直接上代码&#xff1a; …

JDBC之CallableStatement执行存储过程

​ 在前面的一篇文章中&#xff0c;我们学习使用Statement、PreparedStatement来完成对数据表的增删改查。而存储过程作为数据库的重要组成部分&#xff08;痛点&#xff0c;当时学的时候头发都掉了好几根&#x1f62d;&#xff09;&#xff0c;那JDBC是如何执行存储过程呢&…

mysql创建定时任务执行存储过程

存储过程已添加好&#xff1a;https://blog.csdn.net/YXWik/article/details/127283316 1.创建定时器用来执行存储过程函数 create event delete_data on schedule every 10 second do call delete_data();这里的第一行代表的创建名称为delete_data的事件 第二行是执行周期为…

Oracle 定时任务执行存储过程

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 一、创建存储过程一、创建定时任务 一、创建存储过程 CREATE OR REPLACE PROCEDURE TESTCREATE AS --存储过程名称I INT : 1; --变量 BEGINWHILE I < 10 LOOP -…

oracle执行存储过程参数,Oracle 执行存储过程四种方法(带参数 不带参数)

1.如果是命令窗口就用exec 存储过程名&#xff1a; 1 EXEC procedure;--procedure是存储过程名 2.如果是 SQL窗口就用 begin 存储过程名 end; 1 2 3 begin procedure;--procedure是存储过程名 end; 3.如果是程序中调用就用 call 存储过程名 &#xff0c;举个栗子&#xff1…

plsql定时执行存储过程

1、创建定时向表中插入数据的存储过程&#xff0c;名为testJob。 CREATE OR REPLACE PROCEDURE "testJob" AS BEGIN EXECUTE IMMEDIATE INSERT INTO TABLE_HIS SELECT * FROM TABLE_AI; COMMIT; END; 2、使用plsql找到定时器对应的DBMS_Jobs文件夹&#xff0c;…

SQLserver存储过程简单写法与设置定时执行存储过程方法

最近工作中需要写SQLserver的存储过程&#xff0c;第一次使用&#xff0c;简单记录下&#xff0c;以防遗忘。 在SQLserver可视化工具中编写&#xff0c;我的工具如下图&#xff1a; 首先点击你的数据库&#xff0c;找到可编程性&#xff0c;在可编程性里面右击存储过程-->点…

symlink() 函数

查看更多 https://www.yuque.com/docs/share/10f959a4-bd7e-47a9-ad78-11a1310613f3

Install fail! Error: EPERM: operation not permitted, symlink

这个是在安装lodash的时候报错的情况 尝试了下方命令&#xff0c;但是还是有误 在这里可能是因为npm缓存的问题&#xff0c;我先删除了 C:\Users\abc(自己电脑的用户名) 文件夹下的 .npmrc 文件&#xff0c;但是没反应。然后运行了下方的命令&#xff1a; 运行&#xff1a;…

error: eperm: operation not permitted, symlink

查询当前配置的镜像 npm get registry > https://registry.npmjs.org/ 设置成淘宝镜像 npm config set registry http://registry.npm.taobao.org/