【为什么要用多线程?】

传统的图形用户界面应用程序都仅仅有一个运行线程,而且一次仅仅运行一个操作。假设用户从用户界面中调用一个比較耗时的操作,当该操作正在运行时,用户界面一般会冻结而不再响应。这个问题能够用事件处理和多线程来解决。

【Linux有线程的概念吗?】

传统的UNIX系统也支持线程的概念,但一个进程里仅仅同意有一个线程,这样多线程就是多进程。Linux下的Posix线程(pthreads)是一种轻量级的进程的移植性实现,线程的调度由内核完毕,每一个线程都有自己的编号。假设使用线程,整体消耗的系统资源较少,线程间通信也比較easy,在project中推荐使用线程。

【使用多线程有什么优点?】

  1. 提高应用程序的响应速度。这对于开发图形界面程序尤其重要,当一个操作耗时非常长时(比方大批量I/O或大量矩阵变换等CPU密集操作),整个系统都会等待这个操作,程序就不能响应键盘、鼠标、菜单等操作,而使用多线程技术可将耗时长的操作置于一个新的线程,从而避免上述问题。
  2. 使多CPU系统更加有效。当线程数不大于CPU数目时,操作系统能够调度不同的线程执行于不同的CPU上。
  3. 改善程序结构。一个既长又复杂的进程能够考虑分为多个线程,成为独立或半独立的执行部分,这样有利于程序的理解和维护。

【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. 1 #ifndef THREAD_H
  2. 2 #define THREAD_H
  3. 3
  4. 4 #include <QThread>
  5. 5 #include <iostream>
  6. 6
  7. 7 class Thread : public QThread
  8. 8 {
  9. 9 Q_OBJECT
  10. 10 public:
  11. 11 Thread();
  12. 12 void setMessage(QString message);
  13. 13 void stop();
  14. 14
  15. 15 protected:
  16. 16 void run();
  17. 17 void printMessage();
  18. 18
  19. 19 private:
  20. 20 QString messageStr;
  21. 21 volatile bool stopped;
  22. 22 };
  23. 23
  24. 24 #endif // THREAD_H

注:

  • stopped被声明为易失性变量(volatile variable,断电或中断时数据丢失而不可再恢复的变量类型),这是由于不同的线程都须要訪问它,而且我们也希望确保它能在不论什么须要的时候都保持最新读取的数值。假设省略keywordvolatile,则编译器就会对这个变量的訪问进行优化,可能导致不对的结果。

thread.cpp代码

  1. 1 #include "thread.h"
  2. 2 #include <QDebug>
  3. 3
  4. 4 Thread::Thread()
  5. 5 {
  6. 6 stopped = false;
  7. 7 }
  8. 8
  9. 9 void Thread::run()
  10. 10 {
  11. 11 while(!stopped)
  12. 12 {
  13. 13 printMessage();
  14. 14 }
  15. 15 stopped = false;
  16. 16 }
  17. 17
  18. 18 void Thread::stop()
  19. 19 {
  20. 20 stopped = true;
  21. 21 }
  22. 22
  23. 23 void Thread::setMessage(QString message)
  24. 24 {
  25. 25 messageStr = message;
  26. 26 }
  27. 27
  28. 28 void Thread::printMessage()
  29. 29 {
  30. 30 qDebug()<<messageStr;
  31. 31 sleep(1);
  32. 32 }

注:

  • QTread提供了一个terminate()函数,该函数能够再一个线程还在运行的时候就终止它的运行,但不推荐用terminate(),由于terminate()不会立马终止这个线程,该线程何时终止取决于操作系统的调度策略,也就是说,它能够随时停止线程运行而不给这个线程自我清空的机会。更安全的方法是用stopped变量和stop()函数,如样例所看到的。
  • 调用setMessage()让第一个线程每隔1秒打印字母“A”,而让第二个线程每隔1秒打印字母“B”。
  • 线程会由于调用printf()而持有一个控制I/O的锁,多个线程同一时候调用printf()在某些情况下回造成控制台输出堵塞,而用qDebug()作为控制台输出一般不会出现上述问题。

threaddialog.h代码

  1. 1 #ifndef THREADDIALOG_H
  2. 2 #define THREADDIALOG_H
  3. 3
  4. 4 #include <QPushButton>
  5. 5 #include <QDialog>
  6. 6 #include <QCloseEvent>
  7. 7 #include "thread.h"
  8. 8
  9. 9 class ThreadDialog : public QDialog
  10. 10 {
  11. 11 Q_OBJECT
  12. 12
  13. 13 public:
  14. 14 ThreadDialog(QWidget *parent=0);
  15. 15
  16. 16 protected:
  17. 17 void closeEvent(QCloseEvent *event);
  18. 18
  19. 19 private slots:
  20. 20 void startOrStopThreadA();
  21. 21 void startOrStopThreadB();
  22. 22 void close();
  23. 23
  24. 24 private:
  25. 25 Thread threadA;
  26. 26 Thread threadB;
  27. 27 QPushButton *threadAButton;
  28. 28 QPushButton *threadBButton;
  29. 29 QPushButton *quitButton;
  30. 30 };
  31. 31
  32. 32 #endif // THREADDIALOG_H

threaddialog.cpp代码

  1. 1 #include "threaddialog.h"
  2. 2
  3. 3 ThreadDialog::ThreadDialog(QWidget *parent) : QDialog(parent)
  4. 4 {
  5. 5 threadA.setMessage("A");
  6. 6 threadB.setMessage("B");
  7. 7
  8. 8 threadAButton = new QPushButton(tr("Start A"), this);
  9. 9 threadAButton->setGeometry(10, 30, 80, 30);
  10. 10 threadBButton = new QPushButton(tr("Start B"),this);
  11. 11 threadBButton->setGeometry(110, 30, 80, 30);
  12. 12 quitButton = new QPushButton(tr("Quit"), this);
  13. 13 quitButton->setGeometry(210, 30, 80, 30);
  14. 14 quitButton->setDefault(true);
  15. 15
  16. 16 connect(threadAButton, SIGNAL(clicked()), this, SLOT(startOrStopThreadA()));
  17. 17 connect(threadBButton, SIGNAL(clicked()), this, SLOT(startOrStopThreadB()));
  18. 18 connect(quitButton, SIGNAL(clicked()), this, SLOT(close()));
  19. 19 }
  20. 20
  21. 21 void ThreadDialog::startOrStopThreadA()
  22. 22 {
  23. 23 if(threadA.isRunning())
  24. 24 {
  25. 25 threadAButton->setText(tr("Stop A"));
  26. 26 threadA.stop();
  27. 27 threadAButton->setText(tr("Start A"));
  28. 28 }
  29. 29 else
  30. 30 {
  31. 31 threadAButton->setText(tr("Start A"));
  32. 32 threadA.start();
  33. 33 threadAButton->setText(tr("Stop A"));
  34. 34 }
  35. 35 }
  36. 36
  37. 37 void ThreadDialog::startOrStopThreadB()
  38. 38 {
  39. 39 if(threadB.isRunning())
  40. 40 {
  41. 41 threadBButton->setText(tr("Stop B"));
  42. 42 threadB.stop();
  43. 43 threadBButton->setText(tr("Strat B"));
  44. 44 }
  45. 45 else
  46. 46 {
  47. 47 threadBButton->setText(tr("Start B"));
  48. 48 threadB.start();
  49. 49 threadBButton->setText(tr("Stop B"));
  50. 50 }
  51. 51 }
  52. 52
  53. 53 void ThreadDialog::closeEvent(QCloseEvent *event)
  54. 54 {
  55. 55 threadA.stop();
  56. 56 threadB.stop();
  57. 57 threadA.wait();
  58. 58 threadB.wait();
  59. 59 event->accept();
  60. 60 }
  61. 61
  62. 62 void ThreadDialog::close()
  63. 63 {
  64. 64 exit(0);
  65. 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. 1 #include "threaddialog.h"
  2. 2 #include <QApplication>
  3. 3
  4. 4 int main(int argc, char *argv[])
  5. 5 {
  6. 6 QApplication app(argc, argv);
  7. 7 ThreadDialog *threaddialog = new ThreadDialog;
  8. 8 threaddialog->exec();
  9. 9 return app.exec();
  10. 10 }

注:

  • 在GUI程序中,主线程也被称为GUI线程,由于它是唯一一个同意运行GUI相关操作的线程。必须在创建一个QThread之前创建QApplication对象。

Qt多线程学习:创建多线程的更多相关文章

  1. C#多线程学习(一) 多线程的相关概念(转)

    什么是进程?当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源.而一个进程又是由多个线程所组成的. 什么是线程?线程是程序中的一个执行流,每个线程都有自己的专有寄 ...

  2. C#多线程学习(一) 多线程的相关概念

    什么是进程?当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源.而一个进程又是由多个线程所组成的. 什么是线程?线程是程序中的一个执行流,每个线程都有自己的专有寄 ...

  3. [转载]C#多线程学习(一) 多线程的相关概念

    原文地址:http://www.cnblogs.com/xugang/archive/2008/04/06/1138856.html 什么是进程?当一个程序开始运行时,它就是一个进程,进程包括运行中的 ...

  4. java中多线程 - 如何创建多线程

    线程 什么是线程: 线程是进程中的一个实体,是被系统独立调度和分派的基本单位,线程自己不拥有系统资源,只拥有一点儿在运行中必不可少的资源,但它可与同属一个进程的其它线程共享进程所拥有的全部资源 表面上 ...

  5. C#多线程学习(四) 多线程的自动管理(线程池)

    在多线程的程序中,经常会出现两种情况: 一种情况:   应用程序中,线程把大部分的时间花费在等待状态,等待某个事件发生,然后才能给予响应                   这一般使用ThreadPo ...

  6. C++并发与多线程学习笔记--多线程数据共享问题

    创建和等待多个线程 数据和共享问题分析 只读的数据 有读有写 其他案例 共享数据的保护案例代码 创建和等待多个线程 服务端后台开发就需要多个线程执行不同的任务.不同的线程执行不同任务,并返回执行结果. ...

  7. C#多线程学习(五) 多线程的自动管理(定时器)

    Timer类:设置一个定时器,定时执行用户指定的函数.               定时器启动后,系统将自动建立一个新的线程,执行用户指定的函数. 初始化一个Timer对象:  Timer timer ...

  8. 【转】C#多线程学习

    C#多线程学习(一) 多线程的相关概念 什么是进程?当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源.而一个进程又是由多个线程所组成的. 什么是线程?线程是程序 ...

  9. C# 多线程学习总结

    C# 多线程学习总结 C#多线程学习(一) 多线程的相关概念 什么是进程? 当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源.而一个进程又是由多个线程所组成的. ...

  10. C#多线程学习(六) 互斥对象

    如何控制好多个线程相互之间的联系,不产生冲突和重复,这需要用到互斥对象,即:System.Threading 命名空间中的 Mutex 类. 我们可以把Mutex看作一个出租车,乘客看作线程.乘客首先 ...

随机推荐

  1. ps 图片提取线稿方法2种 转

  2. [转]Python文件操作

    前言 这里的“文件”不单单指磁盘上的普通文件,也指代任何抽象层面上的文件.例如:通过URL打开一个Web页面“文件”,Unix系统下进程间通讯也是通过抽象的进程“文件”进行的.由于使用了统一的接口,从 ...

  3. 神奇的linux发行版 tiny core linux

    首先官网在此 http://tinycorelinux.net/ 真正轻量级 名字里带有“tiny”又带有“core”,想必又是一个所谓的“轻量级”发行版. 轻量级我们见多了,debian号称是轻量级 ...

  4. 分析fork后多进程对文件的共享

    fork函数是创建一个新的进程作为原进程的子进程,创建的子进程和父进程存在很多的相似性,首先父子进程的虚拟存储空间的用户空间是相同的,是将父进程的拷贝给子进程.同时父子进程对文件的操作是共享方式.因为 ...

  5. 父视图 使用 UIViewAnimationWithBlocks 时,如何让子视图无动画

    tableView使用 UIViewAnimationWithBlocks 时 上面的cell也会一起出现动画, 所以在设置cell的时候 添加 [UIView performWithoutAnima ...

  6. 第三百四十一天 how can I 坚持

    不好,有点肚子疼,凉肚子了. 今天晚上回来看了个电影<聚焦>,貌似明白了一个道理,任何一份职业,只要认识到了它的价值,那就好好干. 计划又放在脑门后了,上班又闲扯了一天.老季公司招人,让我 ...

  7. 100offer 为专业程序打造的招聘平台

    引用一段100offer的简介 优秀程序员最高效的求职方式 您是名优秀的程序员,很多公司都想邀请您加入,您也有一颗躁动的心.可是,换一份工作对于您,是件机会成本很高的事情.您想一次性看很多个不错的机会 ...

  8. HashSet与HashMap、Hashtable

    (最近在老师叫我们用java去实现LRU算法,了解到要用双链表去做,要用到LinkHashMap去做,但自己对java的几大集合框架并不熟悉,在学习过程了解到了HashMap和HashSet,做个简单 ...

  9. Hibernate入门(2)- 不用配置用注解

    在上一个例子里面,我用的配置文件的方式,这次改成注解. pom.xml 增加了hibernate-commons-annotations和hibernate-annotations <proje ...

  10. 好用的编辑框布局控件TdxLayoutControl

    TdxLayoutControl是编辑框的容器,里面的编辑框可以自动对齐,自由拖放,异常方便.