Qt多线程学习:创建多线程
【为什么要用多线程?】
传统的图形用户界面应用程序都仅仅有一个运行线程,而且一次仅仅运行一个操作。假设用户从用户界面中调用一个比較耗时的操作,当该操作正在运行时,用户界面一般会冻结而不再响应。这个问题能够用事件处理和多线程来解决。
【Linux有线程的概念吗?】
传统的UNIX系统也支持线程的概念,但一个进程里仅仅同意有一个线程,这样多线程就是多进程。Linux下的Posix线程(pthreads)是一种轻量级的进程的移植性实现,线程的调度由内核完毕,每一个线程都有自己的编号。假设使用线程,整体消耗的系统资源较少,线程间通信也比較easy,在project中推荐使用线程。
【使用多线程有什么优点?】
- 提高应用程序的响应速度。这对于开发图形界面程序尤其重要,当一个操作耗时非常长时(比方大批量I/O或大量矩阵变换等CPU密集操作),整个系统都会等待这个操作,程序就不能响应键盘、鼠标、菜单等操作,而使用多线程技术可将耗时长的操作置于一个新的线程,从而避免上述问题。
- 使多CPU系统更加有效。当线程数不大于CPU数目时,操作系统能够调度不同的线程执行于不同的CPU上。
- 改善程序结构。一个既长又复杂的进程能够考虑分为多个线程,成为独立或半独立的执行部分,这样有利于程序的理解和维护。
【Qt中创建线程的方法】
仅仅须要子类化QThread并又一次实现它的run()函数就能够了。run()是个纯虚函数,是线程运行的入口,在run()里出现的代码将会在另外线程中被运行。run()函数是通过start()函数来实现调用的。
【实例】
以下一个样例给出了在应用程序中除了主线程外,还提供了线程A和B。假设单击窗体中的button“Start A”,Qt的控制台就会连续输出字母“A”,此时button“Start A”被刷新为“Stop A”。再单击button“Start B”,控制台会交替输出字母“A”和“B”。假设再单击button“Stop A”,则控制台仅仅输出字母“B”。例如以下图所看到的:
程序结构
thread.h代码
1 #ifndef THREAD_H
2 #define THREAD_H
3
4 #include <QThread>
5 #include <iostream>
6
7 class Thread : public QThread
8 {
9 Q_OBJECT
10 public:
11 Thread();
12 void setMessage(QString message);
13 void stop();
14
15 protected:
16 void run();
17 void printMessage();
18
19 private:
20 QString messageStr;
21 volatile bool stopped;
22 };
23
24 #endif // THREAD_H
注:
- stopped被声明为易失性变量(volatile variable,断电或中断时数据丢失而不可再恢复的变量类型),这是由于不同的线程都须要訪问它,而且我们也希望确保它能在不论什么须要的时候都保持最新读取的数值。假设省略keywordvolatile,则编译器就会对这个变量的訪问进行优化,可能导致不对的结果。
thread.cpp代码
1 #include "thread.h"
2 #include <QDebug>
3
4 Thread::Thread()
5 {
6 stopped = false;
7 }
8
9 void Thread::run()
10 {
11 while(!stopped)
12 {
13 printMessage();
14 }
15 stopped = false;
16 }
17
18 void Thread::stop()
19 {
20 stopped = true;
21 }
22
23 void Thread::setMessage(QString message)
24 {
25 messageStr = message;
26 }
27
28 void Thread::printMessage()
29 {
30 qDebug()<<messageStr;
31 sleep(1);
32 }
注:
- QTread提供了一个terminate()函数,该函数能够再一个线程还在运行的时候就终止它的运行,但不推荐用terminate(),由于terminate()不会立马终止这个线程,该线程何时终止取决于操作系统的调度策略,也就是说,它能够随时停止线程运行而不给这个线程自我清空的机会。更安全的方法是用stopped变量和stop()函数,如样例所看到的。
- 调用setMessage()让第一个线程每隔1秒打印字母“A”,而让第二个线程每隔1秒打印字母“B”。
- 线程会由于调用printf()而持有一个控制I/O的锁,多个线程同一时候调用printf()在某些情况下回造成控制台输出堵塞,而用qDebug()作为控制台输出一般不会出现上述问题。
threaddialog.h代码
1 #ifndef THREADDIALOG_H
2 #define THREADDIALOG_H
3
4 #include <QPushButton>
5 #include <QDialog>
6 #include <QCloseEvent>
7 #include "thread.h"
8
9 class ThreadDialog : public QDialog
10 {
11 Q_OBJECT
12
13 public:
14 ThreadDialog(QWidget *parent=0);
15
16 protected:
17 void closeEvent(QCloseEvent *event);
18
19 private slots:
20 void startOrStopThreadA();
21 void startOrStopThreadB();
22 void close();
23
24 private:
25 Thread threadA;
26 Thread threadB;
27 QPushButton *threadAButton;
28 QPushButton *threadBButton;
29 QPushButton *quitButton;
30 };
31
32 #endif // THREADDIALOG_H
threaddialog.cpp代码
1 #include "threaddialog.h"
2
3 ThreadDialog::ThreadDialog(QWidget *parent) : QDialog(parent)
4 {
5 threadA.setMessage("A");
6 threadB.setMessage("B");
7
8 threadAButton = new QPushButton(tr("Start A"), this);
9 threadAButton->setGeometry(10, 30, 80, 30);
10 threadBButton = new QPushButton(tr("Start B"),this);
11 threadBButton->setGeometry(110, 30, 80, 30);
12 quitButton = new QPushButton(tr("Quit"), this);
13 quitButton->setGeometry(210, 30, 80, 30);
14 quitButton->setDefault(true);
15
16 connect(threadAButton, SIGNAL(clicked()), this, SLOT(startOrStopThreadA()));
17 connect(threadBButton, SIGNAL(clicked()), this, SLOT(startOrStopThreadB()));
18 connect(quitButton, SIGNAL(clicked()), this, SLOT(close()));
19 }
20
21 void ThreadDialog::startOrStopThreadA()
22 {
23 if(threadA.isRunning())
24 {
25 threadAButton->setText(tr("Stop A"));
26 threadA.stop();
27 threadAButton->setText(tr("Start A"));
28 }
29 else
30 {
31 threadAButton->setText(tr("Start A"));
32 threadA.start();
33 threadAButton->setText(tr("Stop A"));
34 }
35 }
36
37 void ThreadDialog::startOrStopThreadB()
38 {
39 if(threadB.isRunning())
40 {
41 threadBButton->setText(tr("Stop B"));
42 threadB.stop();
43 threadBButton->setText(tr("Strat B"));
44 }
45 else
46 {
47 threadBButton->setText(tr("Start B"));
48 threadB.start();
49 threadBButton->setText(tr("Stop B"));
50 }
51 }
52
53 void ThreadDialog::closeEvent(QCloseEvent *event)
54 {
55 threadA.stop();
56 threadB.stop();
57 threadA.wait();
58 threadB.wait();
59 event->accept();
60 }
61
62 void ThreadDialog::close()
63 {
64 exit(0);
65 }
注:
- startOrStopA的逻辑是:当单击A的button时,假设系统推断到有线程A在执行中,就把A的button刷新为“Stop A”,表示能够进行stop A的动作,并停止线程A的执行,再将A的button刷新为“Start A”。否则,假设线程A没有执行,就把button刷新为表示能够执行的“Start A”,启动线程A,然后将Abutton刷新为“Stop A”。
- 当不用Qt设计器时,new一个button出来,须要指定一个父类,比方this,否则执行程序,窗体里没有button。
- new了多个button或控件,须要用setGeometry来确定它们的大小和位置,否则前面的被后面的覆盖,终于看到的是最后一个button。setGeometry的前2个參数是相对于窗体的坐标位置,后两个參数是button的长宽。
- 单击Quit或关闭窗体,就停止全部正在执行的线程,而且在调用函数QCloseEvent::accept()之前等待它们全然结束,这样就能够确保应用程序是以一种原始清空的状态退出的。
- 假设没有62~65行的又一次定义close函数,使进程全然退出。否则点击Quitbutton或叉号退出窗体后,进程依旧驻留在系统里。
main.cpp代码
1 #include "threaddialog.h"
2 #include <QApplication>
3
4 int main(int argc, char *argv[])
5 {
6 QApplication app(argc, argv);
7 ThreadDialog *threaddialog = new ThreadDialog;
8 threaddialog->exec();
9 return app.exec();
10 }
注:
- 在GUI程序中,主线程也被称为GUI线程,由于它是唯一一个同意运行GUI相关操作的线程。必须在创建一个QThread之前创建QApplication对象。
Qt多线程学习:创建多线程的更多相关文章
- C#多线程学习(一) 多线程的相关概念(转)
什么是进程?当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源.而一个进程又是由多个线程所组成的. 什么是线程?线程是程序中的一个执行流,每个线程都有自己的专有寄 ...
- C#多线程学习(一) 多线程的相关概念
什么是进程?当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源.而一个进程又是由多个线程所组成的. 什么是线程?线程是程序中的一个执行流,每个线程都有自己的专有寄 ...
- [转载]C#多线程学习(一) 多线程的相关概念
原文地址:http://www.cnblogs.com/xugang/archive/2008/04/06/1138856.html 什么是进程?当一个程序开始运行时,它就是一个进程,进程包括运行中的 ...
- java中多线程 - 如何创建多线程
线程 什么是线程: 线程是进程中的一个实体,是被系统独立调度和分派的基本单位,线程自己不拥有系统资源,只拥有一点儿在运行中必不可少的资源,但它可与同属一个进程的其它线程共享进程所拥有的全部资源 表面上 ...
- C#多线程学习(四) 多线程的自动管理(线程池)
在多线程的程序中,经常会出现两种情况: 一种情况: 应用程序中,线程把大部分的时间花费在等待状态,等待某个事件发生,然后才能给予响应 这一般使用ThreadPo ...
- C++并发与多线程学习笔记--多线程数据共享问题
创建和等待多个线程 数据和共享问题分析 只读的数据 有读有写 其他案例 共享数据的保护案例代码 创建和等待多个线程 服务端后台开发就需要多个线程执行不同的任务.不同的线程执行不同任务,并返回执行结果. ...
- C#多线程学习(五) 多线程的自动管理(定时器)
Timer类:设置一个定时器,定时执行用户指定的函数. 定时器启动后,系统将自动建立一个新的线程,执行用户指定的函数. 初始化一个Timer对象: Timer timer ...
- 【转】C#多线程学习
C#多线程学习(一) 多线程的相关概念 什么是进程?当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源.而一个进程又是由多个线程所组成的. 什么是线程?线程是程序 ...
- C# 多线程学习总结
C# 多线程学习总结 C#多线程学习(一) 多线程的相关概念 什么是进程? 当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源.而一个进程又是由多个线程所组成的. ...
- C#多线程学习(六) 互斥对象
如何控制好多个线程相互之间的联系,不产生冲突和重复,这需要用到互斥对象,即:System.Threading 命名空间中的 Mutex 类. 我们可以把Mutex看作一个出租车,乘客看作线程.乘客首先 ...
随机推荐
- 无聊之作,RPGdemo制作(一)角色state模式
今天周末,在家无事可做,不知道为什么,突发奇想,想写一个RPG游戏的demo玩玩.所以就记录一下. 第一步要做的是,为以后的角色类写一个state模式的类,考虑到可能以后会用到,所以用模版来实现, / ...
- WARNING: Could not open/create prefs root node Software\JavaSoft\Prefs at root 0x80000002. Windows RegCreateKeyEx(...) returned error code 5
以下是我的程序(取自headfirst Java): import javax.sound.midi.*; public class MiniMiniMusicApp { public static ...
- pick定理:面积=内部整数点数+边上整数点数/2-1
//pick定理:面积=内部整数点数+边上整数点数/2-1 // POJ 2954 #include <iostream> #include <cstdio> #include ...
- Python【基础第三篇】
set集合 s1=set() 集合特点: 访问速度快 没有重复项 collections系列(数据类型容器模块) 一.计数器(Counter) Counter是对字典类型的补充,用于追踪值的出现次数. ...
- docker集成管理工具-shipyard的开发环境搭建笔记
前段时间一直在研究openstack,后来老师告诉我需要用docker容器来搭建hadoop集群,所以就将战场转移到docker上来了,话说docker最近这段时间太火了,但是说实话我觉得应用起来还不 ...
- 20160512关于mac安装caffe的记录
记得2015年在mac系统上安装过一次caffe,非常顺利,但是最近群里许多同学反映mac安装caffe出现了各种问题,同时我也在帮助别人安装caffe的时候也遇到了一些坑,不再像以前这么顺利了.估计 ...
- Cloud_panel
传统基础架构应用程序的系统架构师,云计算应用程序的设计确实是相当有挑战性的工作.体现在应用程序架构师首先要了解云计算环境和传统基础架构的差异并且充分利用云计算平台的一些特点来更好的满足用户需求. 对于 ...
- oracle数据库建表
create or replace directory dumpdir as 'E:\oracle\dumpdir';create temporary tablespace ydxt_temp tem ...
- 第二百二十七天 how can I 坚持
今天去了蟒山,天池,刚去的时候身体有点难受,整天都是那样,回来就好多了,不知道怎么回事. 天池竟然是个人造池,挺大,没有去十三陵,回来都很晚了. 去天池竟然是走的小路,已经关了,不让进,里边玲玲清清的 ...
- 门户级UGC系统的技术进化路线——新浪新闻评论系统的架构演进和经验总结(转)
add by zhj:先收藏了 摘要:评论系统是所有门户网站的核心标准服务组件之一.本文作者曾负责新浪网评论系统多年,这套系统不仅服务于门户新闻业务,还包括调查.投票等产品,经历了从单机到多机再到集群 ...