Qt线程

  • 线程创建方法
  • 示例

线程创建方法

Qt下创建线程的方法有两种:
一种是通过继承QThread,并重写run()函数,在run()函数中,编写线程所做的事情,在需要线程的文件中,创建线程对象,并通过start()函数启动线程,运行run()函数。
另一种是继承QObject类之后,在此类中编写相应的信号和槽函数,在需要线程的类中,创建QThread类的指针和此类的指针,之后,通过函数movetoThread()函数实现线程的转移,关联信号槽后,同过QThread的指针调用start()函数开始线程。

示例

实现的功能是:在下面的界面上,程序运行起初,开始线程按钮和停止线程按钮都是置灰状态,无法使用,点击选择文件路径的按钮之后,弹出一个文件选择对话框,选择的文件路径显示在右侧的TextEdit控件中,此时开始线程按钮可以使用,点击开始线程按钮后,开始线程的按钮变为置灰,停止线程按钮变为可用,PlainTextEdit文本中开始显示所选文件的内容,并不断的显示其内容n遍,在这种状态下直接关闭对话框,即没有停止线程的情况下,关闭对话框,保证程序不奔溃,另一种,可点击停止线程按钮,关闭线程,PlainTextEdit中便不再继续输出文本,然后关闭对话框,程序正常结束。
界面如下图所示:

界面上对象与类的对应关系,如下图:

继承QObject的类mythread的代码:
mythread.h

#ifndef MYTHREAD_H
#define MYTHREAD_H#include <QThread>
#include <QObject>
class QNetworkReply;class MyThread : public QObject
{Q_OBJECT
public:explicit MyThread(QObject *parent = nullptr);void SetNetworkReply(QNetworkReply *reply);
public slots:void replyFinished();
//     void slotTransferNetReply(QNetworkReply *reply);
signals:void signDisplayString( QString str);private:QNetworkReply *m_reply;
};#endif // MYTHREAD_H

mythread.cpp

#include "mythread.h"
#include <QFile>
#include <QDebug>
#include <QThread>MyThread::MyThread(QObject *parent) : QObject(parent)
{m_Statua = false;m_bEndThread = false;
}
MyThread::~MyThread()
{m_Statua = false;m_bEndThread = false;
}void MyThread::slotGetFileContent()
{m_bEndThread = true;while(!m_Statua){QString strFileContent;strFileContent = GetFileContent();if(strFileContent.isEmpty()){return ;}emit signFileContent(strFileContent);       QThread::sleep(1);//不加等待1s会出现线程读数据太快,界面还没来得及写,而导致崩溃}m_bEndThread = false;
}void MyThread::slotFilePath(const QString &strPath)
{m_strFilePath = strPath;
}void MyThread::StopThread()
{m_Statua = true;
}QString MyThread::GetFileContent()
{QString str;QFile file(m_strFilePath);if(!file.exists()){qDebug()<<QString("%1文件不存在").arg(m_strFilePath)<<endl;return str;}if(!file.open(QIODevice::ReadOnly|QIODevice::Text)){qDebug()<<m_strFilePath<<"文件打开失败"<<endl;return str;}str = file.readAll();file.close();//为文件打开后记得关闭return str;
}

界面类dialog的代码:
dialog.h

#ifndef DIALOG_H
#define DIALOG_H#include <QDialog>
#include <QThread>
#include "mythread.h"namespace Ui {class Dialog;
}class Dialog : public QDialog
{Q_OBJECTpublic:explicit Dialog(QWidget *parent = nullptr);~Dialog();void closeEvent(QCloseEvent *event);
signals:void signFilePath(const QString &strPath);
public slots:void slotFileContent(const QString &str);
private slots:void on_pushButton_clicked();void on_pushButtonStart_clicked();void on_pushButtonStop_clicked();private:Ui::Dialog *ui;QThread *thread;MyThread *myThread;
//    QThread thread;
//    MyThread myThread;
};#endif // DIALOG_H

dialog.cpp

#include "dialog.h"
#include "ui_dialog.h"
#include <QFileDialog>
#include <QDebug>Dialog::Dialog(QWidget *parent) :QDialog(parent),ui(new Ui::Dialog)
{ui->setupUi(this);ui->pushButtonStop->setEnabled(false);ui->pushButtonStart->setEnabled(false);myThread = new MyThread;thread = new QThread(this);connect(ui->pushButtonStart,SIGNAL(clicked()),this,SLOT(on_pushButtonStart_clicked()));connect(ui->pushButtonStop,SIGNAL(clicked()),this,SLOT(on_pushButtonStop_clicked()));connect(this,&Dialog::signFilePath,myThread,&MyThread::slotFilePath);
//    connect(this,&Dialog::signFilePath,&myThread,&MyThread::slotFilePath);//   connect(&myThread,&MyThread::signFileContent,this,&Dialog::slotFileContent);
}Dialog::~Dialog()
{delete ui;
}void Dialog::closeEvent(QCloseEvent *event)//防止线程正在运行时,关闭进程导致的崩溃
{if(thread->isRunning()){myThread->StopThread();thread->quit();thread->wait();}
//    if(thread.isRunning())
//    {//        myThread.StopThread();//        while(myThread.m_bEndThread)
//        {//            qDebug()<<"-->closeEvent";
//            QThread::sleep(1);
//        }
//        qDebug()<<"-->closeEvent thread end";
//        thread.quit();
//        thread.wait();
//    }event->accept();
}void Dialog::slotFileContent(const QString &str)
{ui->plainTextEdit->appendPlainText(str);
}void Dialog::on_pushButton_clicked()
{QString strCurPath = QDir::currentPath();QString strFileter = QString("源文件(*.h *.cpp);;文本文件(*.txt);;所有文件(*.*)");//QString strFilePath = QFileDialog::getOpenFileName(this,tr("选择文件"),strCurPath,strFileter);if(strFilePath.isEmpty()){return ;}ui->textEdit->setText(strFilePath);emit signFilePath(strFilePath);ui->pushButton->setEnabled(false);ui->pushButtonStart->setEnabled(true);
}void Dialog::on_pushButtonStart_clicked()
{ui->pushButtonStop->setEnabled(true);ui->pushButtonStart->setEnabled(false);
//    myThread = new MyThread;
//    thread = new QThread(this);myThread->moveToThread(thread);//定义为指针就没有问题
//    myThread.moveToThread(&thread);connect(thread,&QThread::started,myThread,&MyThread::slotGetFileContent);connect(thread,&QThread::finished,myThread,&QObject::deleteLater);connect(myThread,&MyThread::signFileContent,this,&Dialog::slotFileContent);thread->start();
//    connect(&thread,&QThread::started,&myThread,&MyThread::slotGetFileContent);
//    connect(&thread,&QThread::finished,&myThread,&QObject::deleteLater);
//    connect(&myThread,&MyThread::signFileContent,this,&Dialog::slotFileContent);
//    thread.start();
}void Dialog::on_pushButtonStop_clicked()
{if(thread->isRunning()){myThread->StopThread();thread->quit();//定义为变量会出现停止线程的时候没有停止,且加上quit()函数后会卡死,直接关闭对话框会崩溃,提示线程正在运行被关闭了thread->wait();ui->pushButton->setEnabled(true);ui->pushButtonStop->setEnabled(false);}
//    if(thread.isRunning())
//    {//        myThread.StopThread();
//        while(myThread.m_bEndThread)
//        {//            qDebug()<<"-->clos button";
//            QThread::sleep(1);
//        }
//        qDebug()<<"-->clos button thread end";
//        thread.exit();//定义为变量会出现停止线程的时候没有停止,且加上quit()函数后会卡死,直接关闭对话框会崩溃,提示线程正在运行被关闭了
//        //thread.wait();
//        ui->pushButton->setEnabled(true);
//        ui->pushButtonStop->setEnabled(false);
//    }
}

主函数main函数的代码没变:

#include "dialog.h"
#include <QApplication>int main(int argc, char *argv[])
{QApplication a(argc, argv);Dialog w;w.show();return a.exec();
}

初次尝试,第一此定义的QThread对象和继承于QObject的类的对象,没有用指针,但是不知为啥,总是会程序奔溃,最后采用了指针,程序可以正常运行。求一份定义变量的情况下,此程序可以正常运行的指导,望各位同仁不吝赐教。

Qt下继承于QObject创建的线程相关推荐

  1. Win32下利用_beginthread函数创建一个线程

    // Win32_Thread.cpp : 定义控制台应用程序的入口点. //#include "stdafx.h" #include <process.h> /* _ ...

  2. Qt中另一种创建线程的方式

    文章目录 1 Qt中另一种创建线程的方式 1.1 另一种创建线程的方式 1.2 同步型线程的设计 1.3 异步型线程的设计 1 Qt中另一种创建线程的方式 1.1 另一种创建线程的方式 历史的痕迹: ...

  3. 运行错误: QObject: 无法创建子线程, 因为父线程位于不同的线程? 到底是什么鬼?

    运行错误: QObject: 无法创建子线程, 因为父线程位于不同的线程?  到底是什么鬼? (Parent is QThread(0x7ffe4a5a1280), parent's thread i ...

  4. QT解决方案中新建动态链接库工程,且继承于QObject,解决无法生成moc_XXX.cpp文件的问题,解决工程之间的引用问题

    QT解决方案中新建动态链接库工程,且继承于QObject,解决无法生成moc_XXX.cpp文件的问题,解决工程之间的引用问题 参考文章: (1)QT解决方案中新建动态链接库工程,且继承于QObjec ...

  5. QT下信号与槽不在同一个线程中如何connect

    QT下信号与槽不在同一个线程中如何connect

  6. Qt下使用QAxObject进行Word转换为PDF出现错误 error: undefined reference to `QAxObject::QAxObject(QObject*)‘

    Qt下使用QAxObject进行Word转换为PDF出现错误 error: undefined reference to `QAxObject::QAxObject(QObject*)' 首先检查以下 ...

  7. 为什么用线程池?解释下线程池参数?线程池处理流程?阻塞队列的作用?为什么是先添加列队而不是先创建最大线程?线程池中线程复用原理?

    为什么用线程池?解释下线程池参数? 1.降低资源消耗:提高线程利用率,降低创建和销毁线程的消耗. 2.提高响应速度:任务来了,直接有线程可用可执行,而不是先创建线程,再执行. 3.提高线程的可管理性: ...

  8. 继承Thread类创建线程类

    一 点睛 通过继承Thread类创建线程并启动多线程的步骤: 1 定义Thread的子类,并重写该类的run()方法,该run()方法的方法体代表了线程需要完成的任务.因此run()方法称为线程执行体 ...

  9. 高并发场景下创建多少线程才合适?一条公式帮你搞定!!

    来自:冰河技术 创建多少线程合适, 要看多线程具体的应用场景.一般来说,我们可以将程序分为:CPU密集型程序和I/O密集型程序, 而针对于CPU密集型程序和I/O密集型程序,其计算最佳线程数的方法是不 ...

最新文章

  1. 面试官:MySQL 表设计要注意什么?
  2. python气象数据可视化_python grib气象数据可视化
  3. 表单美化-原生javascript和jQuery下拉列表(兼容IE6)
  4. Python-快速排序算法
  5. 从零开始的C++网络编程
  6. html文本设置float,css怎么float(浮动)?
  7. iview-cli 采坑记录
  8. 启迪公交:DRDS助力城市公交系统智能化
  9. linux I/O 栈 预习(上)
  10. 自从安上了“AI”,这些商务经理天天按时下班了
  11. 安全上下文以及FACL
  12. c# 开发项目的过程
  13. jedate change事件监控,使用jedate无法使用change事件
  14. 如何查计算机的网络速度,电脑怎么查网速?教你精准测速的方法
  15. php管理智能dns,智能DNS解析与用户定位调度技术
  16. 福永苹果服务器维修,苹果维修记录iphone app store更新打不开的解决方法
  17. 大学生毕业后的档案问题如何处理
  18. python设置表格格式_python openpyxl表格样式设置
  19. 微信会员卡管理系统:店铺智慧化管理和营销,只需一款软件
  20. 当代最值得收藏的画家作品_当代最具收藏价值的画家:许敬如作品欣赏

热门文章

  1. 深入理解C++中的RVO
  2. Kubernetes 是一个“数据库”吗?
  3. 【开发者成长】喧哗的背后:Serverless 的挑战
  4. 基于Flink的超大规模在线实时反欺诈系统的建设与实践
  5. 彻底弄懂为什么不能把栈上分配的数组(字符串)作为返回值
  6. Redis 混合存储最佳实践指南
  7. 解读NoSQL最新现状和趋势:云NoSQL数据库将成重要增长引擎
  8. 蚂蚁金服CTO程立:做工程要有“拧螺丝”的精神
  9. 当Kubernetes应用遇到阿里分批发布模式
  10. 2017双11技术揭秘—X-DB支撑双11进入分布式数据库时代