Qt多线程学习:创建多线程

【为什么要用多线程?】

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

【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 #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多线程学习:创建多线程,布布扣,bubuko.com

时间: 2024-08-02 02:46:32

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

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

什么是进程?    当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源.而一个进程又是由多个线程所组成的. 什么是线程?    线程是程序中的一个执行流,每个线程都有自己的专有寄存器(栈指针.程序计数器等),但代码区是共享的,即不同的线程可以执行同样的函数. 什么是多线程?    多线程是指程序中包含多个执行流,即在一个程序中可以同时运行多个不同的线程来执行不同的任务,也就是说允许单个程序创建多个并行执行的线程来完成各自的任务. 多线程的好处:    可以提

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

什么是进程?当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源.而一个进程又是由多个线程所组成的. 什么是线程?线程是程序中的一个执行流,每个线程都有自己的专有寄存器(栈指针.程序计数器等),但代码区是共享的,即不同的线程可以执行同样的函数. 什么是多线程?多线程是指程序中包含多个执行流,即在一个程序中可以同时运行多个不同的线程来执行不同的任务,也就是说允许单个程序创建多个并行执行的线程来完成各自的任务. 多线程的好处:可以提高CPU的利用率.在多线程程序中

java多线程学习之——多线程中几种释放锁和不释放锁的操作

在学习java多线程这一块知识的时候,遇到了很多个关于线程锁什么时候释放的问题,现总结几种释放锁和不释放锁的操作如下: 不释放锁: 线程执行同步代码块或同步方法时,程序调用Thread.sleep(Long l).Thread.yield()方法暂停当前线程的执行 线程执行同步代码块时,其它线程调用该线程suspend()方法将该线程挂起,该线程不会释放锁(同步监视器) 尽量避免使用suspend()和resume()来控制线程 释放锁: 当前线程的同步方法.同步代码块执行结束 当前线程的同步方

多线程学习之多线程访问共同资源(队列,多线程,锁机制)实例

模拟场景:main方法为网络请求线程(也叫生产者线程),在网络请求线程中开启四个线程(消费者线程),进行高效处理队列中的共同资源(生产者线程生产的共同资源),等待资源处理完毕,网络请求线程执行结束,响应客户端. 消费者线程体 1 /** 2 * 3 */ 4 package com.benxq.Queue; 5 6 import java.text.SimpleDateFormat; 7 import java.util.Date; 8 import java.util.concurrent.B

iosGCD多线程之创建多线程

喔尽量分成一小节一小节的写.这样也难让大家看的清楚些.我这里有三种创建线程的方法.代码如下 #import "ViewController.h" @interface ViewController () @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; } //当用户点击屏幕,执行线程 -(void)touchesBegan:(NSSet *)touches withEvent:

学习pthreads,多线程的创建和终止

在多CPU多线程的编程中,通过作者的学习发现,pthreads的运用越来越广泛,它是线程的POSIX标准,定义了创建和操作线程的一整套API.环境的配置见上一篇博文,配置好环境后只需要添加#include <pthread.h>,就可以使用pthreads的API了.本文主要介绍一下如何使用pthreads创建多线程,并终止线程.分为三个部分,第一部分给出代码示例,第二部分对代码进行讲解,第三部分给出运行结果. 一.代码示例 本程序创建了5个线程,分别输出Hello World!以及线程编号.

【多线程学习记录一(1)】创建java多线程

1)java多线程的创建方式有三种: 1.继承Thread类 2.实现Runnable接口 3.实现Callable接口 第一种: 1 //继承Thread类 2 class ExtendsThread extends Thread{ 3 public void run(){ 4 //do your stuff 5 } 6 } 7 ExtendsThread e = new ExtendsThread(); 8 e.start(); 第二种: 1 //实现Runnable接口 2 class I

Java多线程学习(吐血超详细总结)

林炳文Evankaka原创作品.转载请注明出处http://blog.csdn.net/evankaka 目录(?)[-] 一扩展javalangThread类 二实现javalangRunnable接口 三Thread和Runnable的区别 四线程状态转换 五线程调度 六常用函数说明 使用方式 为什么要用join方法 七常见线程名词解释 八线程同步 九线程数据传递 本文主要讲了java中多线程的使用方法.线程同步.线程数据传递.线程状态及相应的一些线程函数用法.概述等. 首先讲一下进程和线程

Java多线程学习(详细)

一.进程与线程的区别 进程:每个进程都有独立的代码和数据空间(进程上下文),进程间的切换会有较大的开销,一个进程包含1--n个线程.(进程是资源分配的最小单位) 线程:同一类线程共享代码和数据空间,每个线程有独立的运行栈和程序计数器(PC),线程切换开销小.(线程是cpu调度的最小单位) 线程和进程一样分为五个阶段:创建.就绪.运行.阻塞.终止.     多进程是指操作系统能同时运行多个任务(程序).     多线程是指在同一程序中有多个顺序流在执行. 在java中要想实现多线程,有两种手段,一