QThread与QWidget使用
本文主要内容:
在任务一中,用 四 种方式实现:点击界面按钮,开线程运行一段程序,结果显示在一个Label上。
1. 用不正确的方式得到看似正确的结果
2. 用Qt Manual 和 例子中使用的方法
3. 用一种好用但被Qt开发人员批判的方法
4. 用一种被开发人员强烈推荐,但Qt Manual和例子中只字未提的方法
为了简单起见,本文只讲如何做及其结果是什么,而不讲其原因是什么(估计大家对原因也不会感兴趣,详见: QThread 使用探讨 和 QThread使用方法)。
- 本文只考虑两个线程(即主线程和一个次线程)的情况。
QWidget
- QWidget及其派生类均 不能在次线程中使用或创建
Manual 中的原话:
- The GUI classes, notably QWidget and all its subclasses, are not reentrant. They can only be used from the main thread.
- 因为不允许,所以尝试这么做的,几乎很快都能回头。毕竟signals和slots用起来确实蛮方便
- 但是,回头后,就理解和用对 QThread 了么?
QThread
概念一:QThread 对象本身所依附的线程 和它管理的线程不是同一个线程。
- 前者是主线程
- 后者是次线程
概念二:你在QThread派生类中定义的槽是在主线程而不是在次线程中执行的。
- run 函数是线程的入口点,run内的代码才是在次线程中运行的代码
概念三:除了Manual和Qt例子中给出的用法外,QThread有一种更容易且被推荐的使用方法:
- QThread 应该被看做是操作系统线程的接口或控制点,而不应该包含需要在新线程中运行的代码
- 需要运行的代码应该放到一个QObject的子类中,然后将该子类的对象moveToThread到新线程中。
关于本文的例子
- 为了代码简单,所有例子都是单一的源文件,保存为 main.cpp
你从代码中包含的 #include“main.moc”应该能看出
为了省几行代码,头文件都是直接包含 QtCore 和 QtGui。
- 为了清楚告诉大家槽函数分别是在那个线程运行的,调用了几处 currentThreadId 函数
main 函数中输出主线程IDqDebug()<<"main: "<<QThread::currentThreadId();
run 函数中输出次线程IDqDebug()<<"thread: "<<currentThreadId();
槽函数中输出其在哪个线程中执行qDebug()<<"slots1: "<<currentThreadId();
因为用了qDebug,所以你的pro文件内最好加上 CONFIG+=console
- 同样为了省代码,例子中未考虑线程如何正常结束的问题。
任务一
点击界面按钮,开线程运行一段程序,结果显示在一个Label上。
- 定义一个Widget,上面放置 QPushButton 和 QLabel
- 定义一个Thread,执行我们的代码,然后通知 Widget
第一次尝试
很容易想到方法,代码可以工作,结果正确。但 ... 未必和你想得一样
- Thread 中定义一个slot1函数,接受数据,计算其立方,然后将结果通过信号发出
- Widget 中按钮每点击一次,发出的数据加1,用label接受Thread的信号
- #include <QtCore>
- #include <QtGui>
- class Thread:public QThread
- {
- Q_OBJECT
- public:
- Thread(){}
- public slots:
- void slot1(int v)
- {
- qDebug()<<"slots1: "<<currentThreadId();
- emit sig1(QString::number(v*v*v));
- }
- signals:
- void sig1(const QString& t);
- protected:
- void run()
- {
- qDebug()<<"thread: "<<currentThreadId();
- exec();
- }
- };
- class Widget:public QWidget
- {
- Q_OBJECT
- public:
- Widget():m_label(new QLabel), m_button(new QPushButton("Button")), m_thread(new Thread)
- {
- QVBoxLayout * vbox = new QVBoxLayout(this);
- vbox->addWidget(m_label);
- vbox->addWidget(m_button);
- setLayout(vbox);
- connect(m_button,SIGNAL(clicked()),this,SLOT(onButtonClicked()));
- connect(this,SIGNAL(clicked(int)),m_thread,SLOT(slot1(int)));
- connect(m_thread,SIGNAL(sig1(QString)),m_label,SLOT(setText(QString)));
- m_thread->start();
- }
- signals:
- void clicked(int v);
- private slots:
- void onButtonClicked()
- {
- static int v = 0;
- emit clicked(v);
- v++;
- }
- private:
- QLabel * m_label;
- QPushButton * m_button;
- Thread * m_thread;
- };
- #include "main.moc"
- int main(int argc, char** argv)
- {
- QApplication app(argc, argv);
- qDebug()<<"main: "<<QThread::currentThreadId();
- Widget w;
- w.show();
- return app.exec();
- }
一切工作正常,但看看控制台输出呢?
- main: 3055777552
- thread: 3024481136
- slots1: 3055777552
- slots1: 3055777552
- slots1: 3055777552
- ...
这儿明确告诉你,slot1 是在主线程中执行的。
尝试二
我们试试 Qt Manual和 Qt 例子中采用的解决方案。
槽函数不是在主线程运行么,而run函数不是次线程么?那么我们就:
- 在槽函数中做个标记
- 在run函数中根据标记进行运行
这样以来,尽管槽函数在仍在主线程,但费时的计算代码都在次线程了。
对Thread的类的改造如下(程序其他部分和 尝试一 完全一样):
- class Thread:public QThread
- {
- Q_OBJECT public:
- Thread(){}
- public slots:
- void slot1(int v)
- {
- qDebug()<<"slots1: "<<currentThreadId();
- m_mutex.lock();
- m_vals.enqueue(v);
- m_mutex.unlock();
- }
- signals:
- void sig1(const QString& t);
- protected:
- void run()
- {
- qDebug()<<"thread: "<<currentThreadId();
- while(1) {
- m_mutex.lock();
- if (!m_vals.isEmpty()){
- int v = m_vals.dequeue();
- emit sig1(QString::number(v*v*v));
- }
- m_mutex.unlock();
- }
- }
- private:
- QQueue<int> m_vals;
- QMutex m_mutex;
- };
注意哦,因为 slot 函数在主线程中,而run函数在次线程中,所以二者需要 QMutex 实现对变量的安全访问。如果你认真看过Qt自带的例子,会发现它始终强调 QMutex 的使用。
尝试三
尝试二是"正统"的做法,但如过你用Google搜索过。那么你可能不会选择尝试二,而是会使用下面的方法(其他部分和尝试一 完全一样)
- class Thread:public QThread
- {
- Q_OBJECT public:
- Thread(){ moveToThread(this); }
- ...
这样以来,slot函数确实是在次线程工作的,看看控制台输出
- main: 3056785168
- thread: 3024444272
- slots1: 3024444272
- slots1: 3024444272
- ...
很有意思?不是么,一条 moveToThread(this),移动到自己。然后问题解决了。
- 因为前面说了,QThread 所依附线程 和 它管理的线程不是同一个。
- 这样,其实将自己所依附的线程改为自己所管理的线程了。
o(∩∩)o...哈哈,不要太高兴哦,这个方法看起来比较舒服,但是它是被官方人员强烈批判的用法
尝试四
终于到我想写的代码了,这是Qt线程的开发者建议的使用方式,但很可惜。直到目前(Qt4.7.0),手册和例子中对此都只字为提。
- 我们不子类话QThread了,我们只需要子类话一个QObject,然后将其move到QThread就行了,看代码:
- 不用子类化QThrad了,我们只需要子类话一个 QObject,需要在次线程中工作的代码,直接放到它的槽中
- class Worker:public QObject
- {
- Q_OBJECT
- public:
- Worker(){}
- public slots:
- void slot1(int v)
- {
- qDebug()<<"slots1: "<<QThread::currentThreadId();
- emit sig1(QString::number(v*v*v));
- }
- signals:
- void sig1(const QString& t);
- };
- 因为没有Thread类了,只有Worker类,Widget代码需做点改动
- class Widget:public QWidget
- {
- Q_OBJECT public:
- Widget():m_label(new QLabel), m_button(new QPushButton("Button")), m_worker(new Worker)
- {
- QVBoxLayout * vbox = new QVBoxLayout(this);
- vbox->addWidget(m_label);
- vbox->addWidget(m_button);
- setLayout(vbox);
- QThread * thread = new QThread(this);
- m_worker->moveToThread(thread);
- connect(m_button,SIGNAL(clicked()),this,SLOT(onButtonClicked()));
- connect(this,SIGNAL(clicked(int)),m_worker,SLOT(slot1(int)));
- connect(m_worker,SIGNAL(sig1(QString)),m_label,SLOT(setText(QString)));
- thread->start();
- }
- signals:
- void clicked(int v);
- private slots:
- void onButtonClicked()
- {
- static int v = 0;
- emit clicked(v);
- v++;
- }
- private:
- QLabel * m_label;
- QPushButton * m_button;
- Worker * m_worker;
- };
main 函数还是和尝试一完全一样
控制台输出结果如下
- main: 3056961296
- slots1: 3024616304
- slots1: 3024616304
- ....
一共两个线程,且二者id不同,说明slot在次线程中
恩。这篇文字似乎又不短了,看来任务二要另起一篇了。
QThread与QWidget使用的更多相关文章
- QT 线程的使用(继承QThread)
对于多线程而言,要注意资源的同步和互斥问题,但对于单独的一个线程,则只需要对它的run方法进行重写. 下面实现了一个简单的线程 widget.h文件 #ifndef WIDGET_H #define ...
- 2、QT分析之QPushButton的初始化
原文地址:http://blog.163.com/net_worm/blog/static/127702419201001003326522/ 在简单的QT程序的第二行,声明了一个QPushButto ...
- QT分析之QPushButton的初始化
原文地址:http://blog.163.com/net_worm/blog/static/127702419201001003326522/ 在简单的QT程序的第二行,声明了一个QPushButto ...
- qt之线程
第一种创建: mythread1.h: #ifndef MYTHREAD_H #define MYTHREAD_H #include<QThread> #include<QDebug ...
- QWidget中结束QThread线程
QThread安全结束 protected: void closeEvent(QCloseEvent *event); void closeEvent(QCloseEvent *event) { th ...
- Qt QThread 多线程使用
一.继承QThread 使用方法 1.创建个继承QThread的类. #ifndef MYTHREAD_H #define MYTHREAD_H #include <QObject> #i ...
- Pyqt Smtplib实现Qthread多线程发送邮件
一. smtplib 的介绍 smtplib.SMTP([host[, port[, local_hostname[, timeout]]]]) SMTP类构造函数,表示与SMTP服务器之间的连接 ...
- QThread
#ifndef MYTHREAD_H #define MYTHREAD_H #include<QThread> class MyThread : public QThread { Q_OB ...
- Qt之QThread(深入理解)
简述 为了让程序尽快响应用户操作,在开发应用程序时经常会使用到线程.对于耗时操作如果不使用线程,UI界面将会长时间处于停滞状态,这种情况是用户非常不愿意看到的,我们可以用线程来解决这个问题. 前面,已 ...
随机推荐
- 通过VMware Tools 将主机windows的目录共享给linux虚拟机
之前有写过 本地虚拟机挂载windows共享目录搭建开发环境 这篇,里面讲通过使用samba来实现网络共享 最近发现其实完全不用这么麻烦,VMware tools就可以帮助我们轻松的共享文件夹 这里引 ...
- Spark及其应用场景初探
最近老大让用Spark做一个ETL项目,搭建了一套只有三个结点Standalone模式的Spark集群做测试,基础数据量大概8000W左右.看了官方文档,Spark确实在Map-Reduce上提升了很 ...
- 桶排序-Swift
import Foundation let b:Array = [5,2,3,1,8] var a:NSMutableArray = [] for var i in 0 ..< 11 { a[i ...
- 移动端和web端前端UI库—Frozen UI、WeUI、SUI Mobile
web http://www.pintuer.com/ 拼图 http://www.h-ui.net/ http://www.layui.com/ 很厉害的一个个人产品 http://amazeui ...
- 轻松学习Linux之入门篇
1.Linux概述: 2.Linux优点 3.linux历史待上传 4.linux部分发行版 5.linux政府扶持 本文出自 "李晨光原创技术博客" 博客,谢绝转载!
- [札记]IL经典指令解析之方法调度
call.callvirt和calli指令用于完成方法调用,有何区别呢? 1)call使用静态调度,也就是根据引用类型的静态类型来调度方法.call指令根据引用变量的类型来调用方法,因此通常用于调用非 ...
- keil中如何得知所编译程序所占空间大小?
keil编译后出现Program Size: data=21.0 xdata=0 code=2231. 这表明 data= 21.0 数据储存器内部RAM占用21字节, xdata=0 数据 ...
- LeetCode292:Nim Game
You are playing the following Nim Game with your friend: There is a heap of stones on the table, eac ...
- 用 const 限定类的成员函数
类的成员函数后面加 const,表明这个函数不会对这个类对象的数据成员(准确地说是非静态数据成员)作任何改变. 在设计类的时候,一个原则就是对于不改变数据成员的成员函数都要在后面加 const,而对于 ...
- 可以binidng属性的属性【项目】
1:binding后台bool[]数据以及后台ObservableCollection数据 分别见下面xaml的Visibility和Text的Binding public bool[] Rubber ...