QThread与多线程(比较清楚)
QThread类为我们提供了一种平台无关的管理线程的方式。一个QThread对象管理应用程序中的一个线程,该线程从run()函数开始执行。并且,默认情况下,我们可以在run()函数中通过调用QThread::exec()函数来在当前线程中开启一个事件循环。
而使用QThread开启线程的最常用的方式 就是继承QThread类,重写其run()方法,因为我们刚才就说过,QThread代表的线程就是从run()函数开始运行的。
例如:
class WorkerThread : public QThread
{
Q_OBJECT
void run() Q_DECL_OVERRIDE {
QString result;
/* ... here is the expensive or blocking operation ... */
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()函数返回后退出。又因为我们没有在run()函数中调用exec(),所以该线程中没有运行事件循环。
另一种使用线程的方法,是将要完成的工作封装到一个工作者对象中,然后使用QObject::moveToThread()函数将该对象移动到一个线程对象中。如下:
class Worker : public QObject
{
Q_OBJECT
public slots:
void doWork(const QString ¶meter) {
QString result;
/* ... here is the expensive or blocking operation ... */
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的槽函数会在一个独立的线程中运行。但是,我们可以自由的将Worker的槽函数连接到任意的信号上,任意的对象上,任意的线程中。并且,由于Qt提供的信号和槽连接类型中的queued connections类型,使我们可以安全的跨线程连接信号和槽。
并且,很重要的一点是,QThread对象是存活在创建它的那个线程中,而不是在运行run()函数的新线程中。这意味着,连接到QThread的所有的排队型槽函数都是在旧线程中执行的。因此,如果我们希望我们调用的槽函数也在新线程中执行,就必须使用这种工作对象的方式。新的槽函数 不应该直接实现在QThread的子类中。
还有,当子类化QThread时,要记住的一点是线程对象的构造函数在旧线程中运行,而run()在新线程中运行。所以,如果在这两个函数中都访问了一个成员变量,那么就是在两个不同的线程中访问的,要确保访问的安全性。
下面,我们分别使用这两种方式,通过打印线程id的方法,来看一下它们的区别。
我们先写一个工作者类,继承自QObject:
#ifndef WORKER_H
#define WORKER_H
#include <QObject>
#include <QThread>
#include <QDebug>
class Worker : public QObject
{
Q_OBJECT
public:
explicit Worker(QObject *parent = 0);
public slots:
void start();
};
#endif // WORKER_H
#include "worker.h"
Worker::Worker(QObject *parent) : QObject(parent)
{
}
void Worker::start()
{
qDebug() << "child thread: " << QThread::currentThreadId();
}
我在该类中,只定义了一个start()函数,作为我们线程的执行体,其所做的工作也很简单,只是打印出本线程的id。
接下来,再写一个Controller类,其也继承自QObject,用来控制线程的启动。
#ifndef CONTROLLER_H
#define CONTROLLER_H
#include <QObject>
#include "worker.h"
class Controller : public QObject
{
Q_OBJECT
public:
explicit Controller(QObject *parent = 0);
void start();
signals:
void operate();
private:
QThread workerThread;
};
#endif // CONTROLLER_H
#include "controller.h"
Controller::Controller(QObject *parent) : QObject(parent)
{
Worker *worker = new Worker;
worker->moveToThread(&workerThread);
connect(&workerThread, &QThread::finished, worker, &QObject::deleteLater);
connect(this, &Controller::operate, worker, &Worker::start);
workerThread.start();
}
void Controller::start()
{
emit operate();
}
在这个类中,我们在声明了一个start()函数和一个operate()信号,还有一个QThread对象。然后在构造函数中,实例化一个Worker类对象,再使用moveToThread()函数,将其移动到我们定义的线程对象中;最后,为了启动我们的执行体,我们将operate()信号连接到Worker的start()槽函数上。然后,启动我们的线程对象。
main函数如下:
#include <QCoreApplication>
#include "controller.h"
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
qDebug() << "main thread: " << QThread::currentThreadId();
Controller controller;
controller.start();
return a.exec();
}
我们定义一个Controller的对象,然后调用其start()方法即可。而根据我们在Controller::start()的实现中,发出了operate()信号,该信号又连接到Worker::start(),所以会触发该函数的执行,即我们的线程执行体。
其执行结果如下:
可以看出,子线程的槽函数确实在一个独立的线程中运行。
而如果我们用继承QThread的方法来实现该功能,则会看到不同的结果。
先实现Thread类如下:
#ifndef THREAD_H
#define THREAD_H
#include <QThread>
#include <QDebug>
class Thread : public QThread
{
Q_OBJECT
public:
Thread(QObject *parent = Q_NULLPTR);
public slots:
void Come();
protected:
void run() Q_DECL_OVERRIDE;
};
#endif // THREAD_H
#include "thread.h"
Thread::Thread(QObject *parent)
:QThread(parent)
{
}
void Thread::Come()
{
qDebug() << "child thread: " << QThread::currentThreadId();
}
void Thread::run()
{
exec();
}
我们继承了QThread类,重新了run()方法,这是继承QThread开启线程所必须的一步。另外,我们还定义了一个槽函数,Come(),在该函数中打印了本线程的id。为了让线程不退出,我们在run()函数中调用了QThread::exec()函数,为该线程开启一个事件循环,等待某个信号来触发Come()槽函数。
接下来,再修改Controller类如下:
#ifndef CONTROLLER_H
#define CONTROLLER_H
#include <QObject>
#include "worker.h"
class Controller : public QObject
{
Q_OBJECT
public:
explicit Controller(QObject *parent = 0);
void start();
signals:
void operate();
};
#endif // CONTROLLER_H
#include "controller.h"
#include "thread.h"
Controller::Controller(QObject *parent) : QObject(parent)
{
}
void Controller::start()
{
emit operate();
}
我们只在start()函数中,发送了operate()信号。
再修改main函数如下:
#include <QCoreApplication>
#include "controller.h"
#include "thread.h"
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
qDebug() << "main thread: " << QThread::currentThreadId();
Controller controller;
Thread thread;
QObject::connect(&controller, SIGNAL(operate()), &thread, SLOT(Come()));
thread.start();
controller.start();
return a.exec();
}
我们先定义了Controller和Thread对象,然后将Controller的operate()信号连接到Thread的Come()槽函数上,紧接着启动线程,等待信号。最后调用Controller的start()方法,发送我们定义的信号,该信号又会触发Thread类的槽函数,打印出线程ID。
该种方式的执行结果如下:
可见,线程类Thread中的槽函数并没有运行在run()函数所在的新线程中,而是和main函数在同一个线程中,即创建线程对象的线程。这有时恐怕不是我们想要的。所以,当需要线程中的槽函数完全在另一个新线程中执行时,就需要使用moveToThread()的方法。
---------------------
作者:求道玉
来源:CSDN
原文:https://blog.csdn.net/Amnes1a/article/details/70171519
版权声明:本文为博主原创文章,转载请附上博文链接!
QThread与多线程(比较清楚)的更多相关文章
- [技术博客]采用Qthread实现多线程连接等待
采用Qthread实现多线程连接等待 本组的安卓自动化测试软件中,在测试开始前需要进行连接设备的操作,如下图左侧的按钮 后端MonkeyRunner相关操作的程序中提供了connect() ...
- 在Qt(C++)中使用QThread实现多线程
1. 引言 多线程对于需要处理耗时任务的应用很有用,一方面响应用户操作.更新界面显示,另一方面在"后台"进行耗时操作,比如大量运算.复制大文件.网络传输等. 使用Qt框架开发应用程 ...
- 【QT】子类化QThread实现多线程
<QThread源码浅析> 子类化QThread来实现多线程, QThread只有run函数是在新线程里的,其他所有函数都在QThread生成的线程里.正确启动线程的方法是调用QThrea ...
- QThread创建多线程程序
最近在阅读Qt 5.9 C++开发指南,为了加深对书本上内容的理解,参照书上的讲解尝试写了一些demo,用于以后工作中查阅,如果涉及侵权请告知,实例程序samp13_1 mythread.h #ifn ...
- [ PyQt入门教程 ] PyQt5中多线程模块QThread使用方法
本文主要讲解使用多线程模块QThread解决PyQt界面程序唉执行耗时操作时,程序卡顿出现的无响应以及界面输出无法实时显示的问题.用户使用工具过程中出现这些问题时会误以为程序出错,从而把程序关闭.这样 ...
- QT多线程及通过事件进行通信(通过自定义事件,然后QApplication::postEvent给主界面,我之前用的是信号槽)
可以通过QThread实现跨平台的多线程开发,Qt库负责在特定平台上的特定多线程实现.要采用QThread进行多线程开发,首先需要包含头文件: #include <QThread> 然后需 ...
- Qt多线程-QtConcurrent并行运算高级API
版权声明:若无来源注明,Techie亮博客文章均为原创. 转载请以链接形式标明本文标题和地址: 本文标题:Qt多线程-QtConcurrent并行运算高级API 本文地址:http://tec ...
- PyQt5中Web操作与多线程定时器
1.装载外部网页页面'''用web浏览器控件QWebEngineView控件显示网页PyQt5和Web的交互技术 同时使用python和Web开发程序,混合开发 Python+JavaScript+H ...
- 【QT】子类化QObject+moveToThread实现多线程
往期链接: <QThread源码浅析> <子类化QThread实现多线程> 从往期<QThread源码浅析>可知,在Qt4.4之前,run 是纯虚函数,必须子类化Q ...
随机推荐
- hiredis中异步的实现小结
hiredis中异步的实现小结 原文: http://blog.csdn.net/l1902090/article/details/3858... 时间: 2014-08-15 前言 一般情况下我们使 ...
- Python课程之字典
字典(dict) 一.定义:字典类型在其他语言中又称为map,是一种映射类型,并且{key:value}无序,其关键字必须为不可变类型(如:元组/字符串),在同一个字典中关键字必须互不相同(若出现相同 ...
- 服务器中很多的CLOSE_WAIT
服务器中很多的CLOSE_WAIT,请教各位大虾!!!!!!!!!最近遇到一个问题,工程在LINUX服务器上面跑起来了以后,运行一段时间 就有很多的CLOSE_WAIT链接,多了之后,网站就访问不了了 ...
- Tomcat闲聊第二话
windows下安装Tomcat可以直接下载Installer,需要注意的是,先确保安装了Java虚拟机. 注:默认安装路径为 C:\Program Files\Apache Software Fou ...
- maven项目工程报错:cannot be resolved to a type
1.在本地仓库中,搜索“_maven.repositories”所有匹配项,并彻底删除 2.然后再删除“.lastUpdated”所有匹配项 3.最后再重新在eclipse中执行操作:update d ...
- 如何修改SESSION的生存时间
php中session过期时间设置网上很多人给出了解答:修改php配置文件中的session.gc_maxlifetime.如果想了解更多session回收机制,继续阅读.(本文环境php5.2) 概 ...
- 中移苏研DCOS实践之路完整篇
一.实践背景 1.1现网生产系统存在的问题 在中国移动内部各省市公司,由于技术迭代.设备更新的原因,设备繁杂,有x86server .VMware虚拟机.OpenStack虚拟机以及不同厂商的物理机( ...
- Swift 开源 Linux Ubuntu Install
Swift 开源了,它现在变成跨平台的了,开源后的 Swift 不止能运行在 MAC 和 iOS 平台,现在也可以运行在 Linux 平台了.swift.org 网站上面提供了在 Linux 上面安装 ...
- 推荐个WIN7下小巧的可转录声音的软件-Audio Record Wizard V6.99
之前是XP上用的是 WaveCN 2.0.0.5,但这个软件好久没更新了,不支持WIN7 最终找到了Audio Record Wizard V6.99,尽管没 WaveCN 2.0.0.5好用,但也全 ...
- 【HDU 5305】Friends 多校第二场(双向DFS)
依据题意的话最多32条边,直接暴力的话 2 ^ 32肯定超时了.我们能够分两次搜索时间复杂度降低为 2 * 2 ^ 16 唯一须要注意的就是对眼下状态的哈希处理. 我採用的是 十进制表示法 跑的还是 ...