最近的项目上用到了关于多线程的知识,自己也比较感兴趣,所以就拿了那本《C++ GUI Qt4 编程》来学习。

这本书的第14章是关于多线程的知识,使用的Qt版本是Qt4.x。在下用的是最新的Qt 5.2,所以代码上有一些不兼容,稍加修改就可以运行了。

Qt的多线程简单来说就是继承QThread类,重载run()函数,start()启动线程。首先来看下书上的第一个例子:(修改版的代码已上传,点击下载

class Thread : public QThread
{
Q_OBJECT
public:
Thread(QString message = "", QObject *parent = NULL);
~Thread();
void setMessage(QString);
QString getMessage(); void stop(); protected:
void run(); private:
QString message;
volatile bool stopped;
};

Thread类继承了QThread类,并实现了run函数。stopped变量前面的volatile声明stopped为易失性变量,这样每次读取stopped时都是最新的值。

继续看Thread类的实现:

Thread::Thread(QString message, QObject *parent) :
stopped(false)
, QThread(parent)
, message(message)
{
} Thread::~Thread()
{
this->stop();
this->wait();
qDebug() << this;
} void Thread::setMessage(QString message)
{
this->message = message;
} QString Thread::getMessage()
{
return this->message;
} void Thread::stop()
{
stopped = true;
} void Thread::run()
{
while (!stopped)
std::cerr << qPrintable(message);
stopped = false;
std::cerr << std::endl;
}

初始化时将stopped设置为false,run函数中持续检查stopped的值,为true时才退出。

Dialog::Dialog(QWidget *parent)
: QDialog(parent)
{
QPushButton *buttonQuit = new QPushButton(QString::fromLocal8Bit("Quit"));
connect(buttonQuit, &QPushButton::clicked, this, &Dialog::close); QBoxLayout *layout = new QBoxLayout(QBoxLayout::LeftToRight, this); QStringList list = QString("ABCDEFGHIJKLMN").split("",QString::SkipEmptyParts); foreach (QString name, list)
{
Thread *thread = new Thread(name, this);
QPushButton *button = new QPushButton(QString("Start ")+name, this);
mappingTable.insert(button, thread);
connect(button, &QPushButton::clicked, this, &Dialog::startOrStopThread);
layout->addWidget(button);
} layout->addWidget(buttonQuit);
this->setLayout(layout);
} void Dialog::startOrStopThread()
{
QPushButton *buttonNow = dynamic_cast<QPushButton*>(sender());
Thread *threadNow = (Thread*)mappingTable[buttonNow]; if (threadNow == NULL) return; if(threadNow->isRunning())
{
threadNow->stop();
buttonNow->setText( buttonNow->text().replace(QString("Stop"),QString("Start")) );
}
else
{
threadNow->start();
buttonNow->setText( buttonNow->text().replace(QString("Start"),QString("Stop")) );
}
}

在Dialog界面类中,将button与thread实现一一对应的连接,在槽函数中就可以方便的找到对应的线程了。其中mappingTable是QMap<QObject*, QObject*>类型的。

这样就可以方便的实现多个线程的修改,如下图:

另外,第四个例子对我也很有启发:

TransactionThread::TransactionThread(QObject *parent) :
QThread(parent)
{
start();
} TransactionThread::~TransactionThread()
{
{
QMutexLocker locker(&mutex); while (!transactions.isEmpty())
delete transactions.dequeue(); transactionCondition.wakeOne();
} wait();
} void TransactionThread::addTransaction(Transaction *transaction)
{
QMutexLocker locker(&mutex);
transactions.enqueue(transaction);
transactionCondition.wakeOne();
} void TransactionThread::run()
{
Transaction *transaction = ;
QImage oldImage; forever
{
{
QMutexLocker locker(&mutex); if (transactions.isEmpty())
transactionCondition.wait(&mutex); if (transactions.isEmpty())
break; transaction = transactions.dequeue();
oldImage = currentImage;
} emit transactionStarted(transaction->message(), );
QImage newImage = transaction->apply(oldImage);
delete transaction; {
QMutexLocker locker(&mutex);
currentImage = newImage; if (transactions.isEmpty())
emit allTransactionsDone();
}
}
} void TransactionThread::setImage(const QImage& image)
{
QMutexLocker locker(&mutex);
currentImage = image;
} QImage TransactionThread::getImage()
{
QMutexLocker locker(&mutex);
return currentImage;
}

以上为线程实现的关键代码。在读取和写入从线程与主线程共享的变量时,都要使用mutex互斥变量。使用QMutexLocker locker(&mutex)也更方便,在构造是lock,析构时unlock,临时变量超过了作用域自然被析构,不得不说实现者的方法很巧妙啊。至于transactionCondition.wait(&mutex)则是等待条件。当事务队列为空时,等待事务加入,或者析构。加入事务时唤醒即可,即transactionCondition.wakeOne()。

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

  1. [转] Qt 多线程学习

    Qt 多线程学习 转自:http://www.cnblogs.com/IT-BOY/p/3544220.html 最近的项目上用到了关于多线程的知识,自己也比较感兴趣,所以就拿了那本<C++ G ...

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

    [为什么要用多线程?] 传统的图形用户界面应用程序都仅仅有一个运行线程,而且一次仅仅运行一个操作.假设用户从用户界面中调用一个比較耗时的操作,当该操作正在运行时,用户界面一般会冻结而不再响应.这个问题 ...

  3. Qt多线程学习-用例子来理解多线程(转),这个是我看过最好的文章,总结很详细(感觉exec()的作用就是保持线程不退出,这样方便随时处理主线程发来的信号,是一种非常别致的思路)good

    01 class MThread :public QThread 02 { 03 public: 04     MThread(); 05     ~MThread(); 06     virtual ...

  4. Qt多线程学习-用例子来理解多线程

    文章出处:DIY部落(http://www.diybl.com/course/3_program/c/c_js/20090303/157373_3.html) POINT 1:QThread类的实例与 ...

  5. QT多线程学习

    一.想要使用Qthread必须先创建,继承Qthread的类. #ifndef THREADTEST_H #define THREADTEST_H #include <QThread> # ...

  6. Qt 多线程和网络编程学习

    一,Qt多线程类学习 QThread类,开始一个新的线程就是开始执行重新实现QThread::run(),run()是默认现实调用exec(),QThread::start()开始线程的执行,run( ...

  7. QT入门学习笔记2:QT例程

    转至:http://blog.51cto.com/9291927/2138876 Qt开发学习教程 一.Qt开发基础学习教程 本部分博客主要根据狄泰学院唐老师的<QT实验分析教程>创作,同 ...

  8. 【QT】 Qt多线程的“那些事”

    目录 一.前言 二.QThread源码浅析 2.1 QThread类的定义源码 2.2 QThread::start()源码 2.3 QThreadPrivate::start()源码 2.4 QTh ...

  9. Java多线程学习笔记

    进程:正在执行中的程序,其实是应用程序在内存中运行的那片空间.(只负责空间分配) 线程:进程中的一个执行单元,负责进程汇总的程序的运行,一个进程当中至少要有一个线程. 多线程:一个进程中时可以有多个线 ...

随机推荐

  1. 7zip 命令行

    转自 http://www.cnblogs.com/langlang/archive/2010/12/01/1893866.html 7z.exe 是 7-Zip 的命令行版本.7z.exe 使用 7 ...

  2. iOS Core data多线程并发访问的问题

    大家都知道Core data本身并不是一个并发安全的架构:不过针对多线程访问带来的问题,Apple给出了很多指导:同时很多第三方的开发者也贡献了很多解决方法.不过最近碰到的一个问题很奇怪,觉得有一定的 ...

  3. MAC 13信道

    房东的路由器一直连不上,手机却能连上,前天设置了13信道,后来又忘了,最后找到个连接WIFI的方法,在网络偏好设置里选择向导,诊断中可以连上wifi.

  4. Mac和iOS开发资源汇总

    小引 本文主要汇集一些苹果开发的资源,会经常更新,建议大家把这篇文章单独收藏(在浏览器中按command+D). 今天(2013年7月19日)收录了许多中文网站和博客.大家一定要去感受一下哦. 如果大 ...

  5. 总结:Unity3D游戏上线后的流程回顾

    原地址:http://unity3d.9tech.cn/news/2014/0127/39748.html 首先.unity 灯光烘焙 :Unity 3d FBX模型导入.选项Model 不导入资源球 ...

  6. 【redis】05Redis的常用命令及高级应用

    Redis常用命令     Redis提供了非常丰富的命令,对数据库和个中数据类型进行操作, 这些命令呢,可以在Linux终端使用. 分为两大类的命令,一种是键值相关的命令,一种是服务器相关的命令, ...

  7. React-Native学习指南

    React-Native学习指南 本指南汇集React-Native各类学习资源,给大家提供便利.指南正在不断的更新,大家有好的资源欢迎Pull Requests! 同时还有Awesome React ...

  8. Struts 2知识回顾----拦截器(Intercept)总结

    什么是Struts 2拦截器? 从软件构架上来说,拦截器是实现了面向方面编程的组件.它将影响了多个业务对象的公共行为封装到一个个可重用的模块,减少了系统的重复代码,实现功能的高度内聚,确保了业务对象的 ...

  9. CF 86D Powerful array

    离线+分块 将n个数分成sqrt(n)块. 对所有询问进行排序,排序标准:       1. Q[i].left /block_size < Q[j].left / block_size (块号 ...

  10. 6.5 THUSC 考试题解

    QAQ 由于并没有数据,而且没有A掉的是提交答案题目,所以并没有修改 QAQ 只能放题解了,代码还没有拿到,不过在清华听了一波习题讲评的安利 第一题 成绩单 先说暴力分 对于单调序列来说最优决策一定是 ...