qt QThread

QThread类提供了一个平台无关的方式来管理线程。

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

可以使用worker-object通过QObject::moveToThread将它们移动到线程。
class Worker :
public QObject
{
Q_OBJECT

public slots:
void
doWork(const QString &parameter) {
// ...
emit
resultReady(result);
}

signals:
void resultReady(const
QString &result);
};

class Controller : public QObject
{

Q_OBJECT

QThread 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槽中的代码将在一个单独的线程中执行,然而,可以将(来自任何对象、在任何线程中)任何信号与该槽自由地连接,在不同的线程里连接信号和槽也是安全的,这要归功于一个叫排队的连接机制(queued
connections)。
另一种使代码运行在一个单独的线程中的方法,是子类化QThread中并重新实现的run()。
例如:

class WorkerThread : public QThread
{
Q_OBJECT

void
run() Q_DECL_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()。

注意一个线程实例位于实例化它的旧线程中,而非调用run()的新线程中,这意味着所有线程的排队槽将在旧线程中执行。因此,开发人员希望在新线程调用槽必须使用worker-object方法,新槽不应直接在子类化QThread中来实现。


当子类化QThread时,请记住,构造函数在旧线程中执行,然而run()在新线程中执行。如果一个成员变量的访问来自两个函数,然后从两个不同的线程访问变量,需要检查这样做是否安全。

注:用在不同的线程中的对象进行交互时必须小心。详见同步线程(Synchronizing Threads)。

管理线程

QThread会通知你触发了一个信号当线程started()和finished()时,或者使用isFinished()和isRunning()来查询线程的状态。


可以通过调用exit()或quit()来停止线程。在极端情况下,你可能要强行terminate()一个执行线程。但是,这样做是危险的。请阅读文档查看terminate()和setTerminationEnabled()的详细信息。

从Qt4.8起,可以释放运行刚刚结束的线程对象,通过连接finished()信号到QObject::deleteLater()。

使用wait()来阻塞调用的线程,直到其他线程执行完毕(或者直到指定的时间过去)。

QThread中还提供了静态的、平台独立的休眠功能:sleep()、msleep()、usleep()允许秒,毫秒和微秒来区分,这些函数在Qt5.0中被设为public。


注意:一般情况下,wait()和sleep()函数应该不需要,因为Qt是一个事件驱动型框架。而不是wait(),关心监听信号finished()。取代sleep(),可以考虑使用QTimer。


静态函数currentThreadId()和currentThread()返回标识当前正在执行的线程。前者返回该线程的平台特定的ID,后者返回一个线程指针。


要设置线程的名称,可以在启动线程之前调用setObjectName()。如果不调用setObjectName(),线程的名称将是线程对象的运行时类型(上例中“WorkerThread”,因为这是QThread子类的类名)。请注意,基于Windows的构建版本目前不可用。

时间: 2024-08-06 10:11:09

qt QThread的相关文章

Qt - QThread(翻译帮助文档)

QThread Class 详细描述 QThread 类提供一个平台无关的方法来管理线程. 一个QThread对象管理一个程序中的控制线程.QThread在run()中开始执行任务.默认地,run()通过调用exec()开始事件循环,并且在线程中运行Qt事件循环. 你可通过使用QObject::moveToThread()将worker对象移到线程中来使用worker对象. class Worker : public QObject { Q_OBJECT public slots: void d

Qt QThread必须要了解的几个函数

概述 如果想对Qt中的QThread有个更加深刻的了解,必须要知道这几个重要的函数,现在就一一介绍下. 函数介绍 属性 返回值 函数体 功能 static QThread * QThread::currentThread() 返回当前线程的指针,静态函数. static Qt::HANDLE QThread::currentThreadId() 返回当前线程的句柄,静态函数 static bool QThread::isFinished() const 如果线程执行结束,返回true,否则返回f

QThread多线程编程经典案例分析(三种方法,解释了为什么使用moveToThread的根本原因,即为了避免调用QThread::exec() )

传统的图形界面应用程序都只有一个线程执行,并且一次执行一个操作.如果用户调用一个比较耗时的操作,就会冻结界面响应. 一个解决方法是按照事件处理的思路: 调用 Void QApplication::processEvents() 或 void QApplication::processEvents ( int maxtime ) 来强迫事件循环进行,但是这种做法是有潜在风险的.按照QCoreApplication:processEvents()可能会引起递归,导致栈溢出崩溃的说法,当主线程在某个槽

QThread多线程编程经典案例分析

传统的图形界面应用程序都只有一个线程执行,并且一次执行一个操作.如果用户调用一个比较耗时的操作,就会冻结界面响应. 一个解决方法是按照事件处理的思路: 调用 Void QApplication::processEvents() 或 void QApplication::processEvents ( int maxtime ) 来强迫事件循环进行,但是这种做法是有潜在风险的.按照QCoreApplication:processEvents()可能会引起递归,导致栈溢出崩溃的说法,当主线程在某个槽

Threads Events QObjects

Events and the event loop Being an event-driven toolkit, events and event delivery play a central role in Qt architecture. In this article we'll not give a comprehensive coverage about this topic; we'll instead focus on some thread-related key concep

QT之深入理解QThread

QT之深入理解QThread 理解QThread之前需要了解下QThread类,QThread拥有的资源如下(摘录于QT 5.1 帮助文档): 在以上资源中,本文重点关注槽:start():信号:started().finished():受保护的方法:run().exec(): 理解QThread QThread与通常所熟知的线程(thread)有很大出入,在面向过程的语言中,我们建立一个线程的同时会传入一个函数名,这个函数名代表该线程要执行的具体代码(如图 1 所示). 图 1. 我们通常所理

Qt线程(2) QThread中使用WorkObject

一般继承QThread的WorkThread都会在重载的run()中创建临时的WorkObject,这样能确定这个WorkObject在该thread中使用 那如果这个WorkObject是个Singleton是种什么情况呢? 方式2:在QThread中使用WorkObject 1.创建WorkObject 1 #include <QObject> 2 #include <QDebug> 3 #include <QThread> 4 5 #define DEBUG(x)

Qt 线程基础(QThread、QtConcurrent等) 2

使用线程 基本上有种使用线程的场合: 通过利用处理器的多个核使处理速度更快. 为保持GUI线程或其他高实时性线程的响应,将耗时的操作或阻塞的调用移到其他线程. 何时使用其他技术替代线程 开发人员使用线程时需要非常小心.启动线程是很容易的,但确保所有共享数据保持一致很难.遇到问题往往很难解决,这是由于在一段时间内它可能只出现一次或只在特定的硬件配置下出现.在创建线程来解决某些问题之前,应该考虑一些替代的技术 : 替代技术 注解 QEventLoop::processEvents() 在一个耗时的计

Qt之QThread(深入理解)

简述 前面,我们介绍了QThread常用的两种方式: worker-object 子类化QThread 下面,我们首先来看看子类化QThread在日常中的应用. 简述 子类化QThread 在主线程中更新UI 正常结束线程 更多参考 一般情况下,QThread进行耗时操作的同时会与UI进行交互,比如:显示进度.旋转等待...进行友好型的交互,让用户知道当前的操作. 子类化QThread 我们以更新进度条为例,来模拟一个耗时操作,并分享我们有可能在此过程中遇到的问题及解决办法. 首先,我们定义一个