Qt下继承于QObject创建的线程
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创建的线程相关推荐
- Win32下利用_beginthread函数创建一个线程
// Win32_Thread.cpp : 定义控制台应用程序的入口点. //#include "stdafx.h" #include <process.h> /* _ ...
- Qt中另一种创建线程的方式
文章目录 1 Qt中另一种创建线程的方式 1.1 另一种创建线程的方式 1.2 同步型线程的设计 1.3 异步型线程的设计 1 Qt中另一种创建线程的方式 1.1 另一种创建线程的方式 历史的痕迹: ...
- 运行错误: QObject: 无法创建子线程, 因为父线程位于不同的线程? 到底是什么鬼?
运行错误: QObject: 无法创建子线程, 因为父线程位于不同的线程? 到底是什么鬼? (Parent is QThread(0x7ffe4a5a1280), parent's thread i ...
- QT解决方案中新建动态链接库工程,且继承于QObject,解决无法生成moc_XXX.cpp文件的问题,解决工程之间的引用问题
QT解决方案中新建动态链接库工程,且继承于QObject,解决无法生成moc_XXX.cpp文件的问题,解决工程之间的引用问题 参考文章: (1)QT解决方案中新建动态链接库工程,且继承于QObjec ...
- QT下信号与槽不在同一个线程中如何connect
QT下信号与槽不在同一个线程中如何connect
- Qt下使用QAxObject进行Word转换为PDF出现错误 error: undefined reference to `QAxObject::QAxObject(QObject*)‘
Qt下使用QAxObject进行Word转换为PDF出现错误 error: undefined reference to `QAxObject::QAxObject(QObject*)' 首先检查以下 ...
- 为什么用线程池?解释下线程池参数?线程池处理流程?阻塞队列的作用?为什么是先添加列队而不是先创建最大线程?线程池中线程复用原理?
为什么用线程池?解释下线程池参数? 1.降低资源消耗:提高线程利用率,降低创建和销毁线程的消耗. 2.提高响应速度:任务来了,直接有线程可用可执行,而不是先创建线程,再执行. 3.提高线程的可管理性: ...
- 继承Thread类创建线程类
一 点睛 通过继承Thread类创建线程并启动多线程的步骤: 1 定义Thread的子类,并重写该类的run()方法,该run()方法的方法体代表了线程需要完成的任务.因此run()方法称为线程执行体 ...
- 高并发场景下创建多少线程才合适?一条公式帮你搞定!!
来自:冰河技术 创建多少线程合适, 要看多线程具体的应用场景.一般来说,我们可以将程序分为:CPU密集型程序和I/O密集型程序, 而针对于CPU密集型程序和I/O密集型程序,其计算最佳线程数的方法是不 ...
最新文章
- 面试官:MySQL 表设计要注意什么?
- python气象数据可视化_python grib气象数据可视化
- 表单美化-原生javascript和jQuery下拉列表(兼容IE6)
- Python-快速排序算法
- 从零开始的C++网络编程
- html文本设置float,css怎么float(浮动)?
- iview-cli 采坑记录
- 启迪公交:DRDS助力城市公交系统智能化
- linux I/O 栈 预习(上)
- 自从安上了“AI”,这些商务经理天天按时下班了
- 安全上下文以及FACL
- c# 开发项目的过程
- jedate change事件监控,使用jedate无法使用change事件
- 如何查计算机的网络速度,电脑怎么查网速?教你精准测速的方法
- php管理智能dns,智能DNS解析与用户定位调度技术
- 福永苹果服务器维修,苹果维修记录iphone app store更新打不开的解决方法
- 大学生毕业后的档案问题如何处理
- python设置表格格式_python openpyxl表格样式设置
- 微信会员卡管理系统:店铺智慧化管理和营销,只需一款软件
- 当代最值得收藏的画家作品_当代最具收藏价值的画家:许敬如作品欣赏
热门文章
- 深入理解C++中的RVO
- Kubernetes 是一个“数据库”吗?
- 【开发者成长】喧哗的背后:Serverless 的挑战
- 基于Flink的超大规模在线实时反欺诈系统的建设与实践
- 彻底弄懂为什么不能把栈上分配的数组(字符串)作为返回值
- Redis 混合存储最佳实践指南
- 解读NoSQL最新现状和趋势:云NoSQL数据库将成重要增长引擎
- 蚂蚁金服CTO程立:做工程要有“拧螺丝”的精神
- 当Kubernetes应用遇到阿里分批发布模式
- 2017双11技术揭秘—X-DB支撑双11进入分布式数据库时代