qt创建多线程的几种方法

一、继承QThread
继承QThread,这应该是最常用的方法了。我们可以通过重写虚函数void QThread::run ()实现我们自己想做的操作,实现新建线程的目的。前面已经介绍了Qthread,这里就不重复了。


这种方法,我们每一次要新建一个线程都需要继承Qthread,实现一个新的类,有点不太方便。但是相对于Qrunnable,这种方法的好处就是我们可以直接调用对象的start()函数启动线程,而Qrunnable必须借助QthreadPool。


 


二、继承QRunnable
Qrunnable是所有可执行对象的基类。我们可以继承Qrunnable,并重写虚函数void QRunnable::run () 。我们可以用QThreadPool让我们的一个QRunnable对象在另外的线程中运行,如果autoDelete()返回true(默认),那么QThreadPool将会在run()运行结束后自动删除Qrunnable对象。可以调用void QRunnable::setAutoDelete ( bool autoDelete )更改auto-deletion标记。需要注意的是,必须在调用QThreadPool::start()之前设置,在调用QThreadPool::start()之后设置的结果是未定义的。


下面是我写的简单的例子:


class Runnable:publicQRunnable


{


       //Q_OBJECT   注意了,Qrunnable不是QObject的子类。


public:


       Runnable();


       ~Runnable();


 


       voidrun();


protected:


private:


};


 


 


Runnable::Runnable():QRunnable()


{


 


}


 


Runnable::~Runnable()


{


       cout<<"~Runnable()"<<endl;


}


 


void Runnable::run()


{


       cout<<"Runnable::run()thread :"<<QThread::currentThreadId()<<endl;


       cout<<"dosomething ...."<<endl;


}


int main(int argc, char *argv[])


{


       QCoreApplication a(argc, argv);


       cout<<"mainthread :"<<QThread::currentThreadId()<<endl;


       Runnable runObj;


       QThreadPool::globalInstance()->start(&runObj);


       returna.exec();


}


 


 


由结果可看出,run()确实是在不同于主线程的另外线程中运行的,而且在运行结束后就调用了析构函数,因为默认是可以自动被销毁的。


我们可以对同一个对象多次调用QThreadPool::start(),如果是可以自动被销毁的,Qrunnable对象会在最后一个线程离开了run函数之后才被销毁的。


int main(int argc, char *argv[])


{


       QCoreApplication a(argc, argv);


       cout<<"mainthread :"<<QThread::currentThreadId()<<endl;


       Runnable runObj;


       QThreadPool::globalInstance()->start(&runObj);


       QThreadPool::globalInstance()->start(&runObj);


       QThreadPool::globalInstance()->start(&runObj);      


       returna.exec();


}


我三次调用QThreadPool::globalInstance()->start(&runObj);,但是在三次都执行完之后才运行析构函数。






这种新建线程的方法的最大的缺点就是:不能使用Qt的信号—槽机制,因为Qrunnable不是继承自QObject。所以我们要想知道线程是否运行结束或获取运行结果可能会比较麻烦。还有就是我们不能直接调用run()启动线程,必须借助于QthreadPool。


但是这种方法的好处就是,可以让QThreadPool来管理线程,QThreadPool会自动的清理我们新建的Qrunnable对象。


 


三、使用moveToThread 
首先我们必须实现继承QObject的一个类,实现我们想要的功能。


class Worker:publicQObject


{


       Q_OBJECT


public:


       Worker();


       ~Worker();


 


protected slots:


       voidfun1();


   void fun2();


private:


};


Worker::Worker():QObject()


{        }


 


Worker::~Worker()


{     }


 


void Worker::fun1()


{


       cout<<"Worker::fun1()  thread : "<<QThread::currentThreadId()<<endl;


}


 


接着创建一个对象,并调用:moveToThread ( QThread * targetThread ),让对象在新的线程中运行。


int main(int argc, char *argv[])


{


       QCoreApplication a(argc, argv);


       cout<<"mainthread :"<<QThread::currentThreadId()<<endl;


       QThread thread;


       Worker work;


       thread.start();              //注意记得启动线程


       work.moveToThread(&thread);


//由于不能直接调用worker


//的函数,所以一般用信号触发调用


QTimer::singleShot(0,&work,SLOT(fun1()));  


QTimer::singleShot(0,&work,SLOT(fun1()));  


       returna.exec();


}


这样就能让fun1()和fun2()都运行在thread线程中了。


 


 


需要注意的是:在work 的函数结束运行前,thread不能被析构。Thread的生命期不能小于work。否则的话程序就好崩掉了。


像下面的代码肯定是不行的。


void Dialog::startWork()


{


       QThread thread;


      Worker*work = new Worker;


       thread.start();


       work->moveToThread(&thread);


       QTimer::singleShot(0,work,SLOT(fun1()));


       QTimer::singleShot(0,work,SLOT(fun2()));


}


所以thread 必须是new出来的。但是这样的话,就感觉有点麻烦,我们要同时管理thread和work,因为都是new 出来,我们需要负责清理。为了避免这样的麻烦,我想到的方法是,在work类中添加一个QThread成员。


class Worker:publicQObject


{


       Q_OBJECT


public:


       Worker();


       ~Worker();


 


       protectedslots:


              voidfun1();


              voidfun2();


private:


       QThread m_thread;


};


 


Worker::Worker():QObject()


{


       m_thread.start();


       this->moveToThread(&m_thread);


}


这样我们在用的时候只需要newwork就行了。


 


四、使用QtConcurrent::run
其实前面也有用到QtConcurrent::run启动新线程了。QtConcurrent命名空间提供了很多方法可以实现并发编程,这个以后再深入探讨了,这里只是大概讲一下启动线程。还是用上面的worker代码作为例子:


void Worker::start()


{


       QtConcurrent::run(this,&Worker::fun1);


       QtConcurrent::run(this,&Worker::fun2);


}


 


 


QtConcurrent::run是个模板函数,有很多种形式,我们也可以让全局的函数允许在另外的线程中。


void printMes(char*mes)


{


       cout<<"pprintMes(char*mes)  thread : "<<QThread::currentThreadId()<<endl;


       cout<<mes<<endl;


}


 


int main(int argc, char *argv[])


{


       QCoreApplication a(argc, argv);


       cout<<"mainthread :"<<QThread::currentThreadId()<<endl;


       char *mes= "hello world";


       QtConcurrent::run(printMes,mes);


       returna.exec();


}


 






 


目前所知的新建线程的方法就大概这些了,希望对大家有用,可能还要别的,以后再继续学习了

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Qt Creator 多线程读取文件到程序显示 利用QT Creator多任务读取一个文档到程序里 为了防止直接读取文件里的内容太大而发生卡顿,于是多线程读取将更高效的解决这个问题。 效果图如下: 其中pro文件无需改动,默认就好,头文件h里面的内容为 #ifndef MAINWINDOW_H #define MAINWINDOW_H #include #include #include QT_BEGIN_NAMESPACE namespace Ui { class MainWindow; } QT_END_NAMESPACE class MyObj; class MyObj : public QObject { Q_OBJECT public: MyObj(); //新的线程 signals: void toLine(QString line); private slots: void doWork(); }; class MainWindow : public QMainWindow { Q_OBJECT public: MainWindow(QWidget *parent = nullptr); ~MainWindow(); private slots: void appendText(QString); //定义一个槽 private: Ui::MainWindow *ui; QThread *t; MyObj *obj; }; #endif // MAINWINDOW_H 而MAIN主文件的内容为了防止中文乱码做了如下修改: #include "mainwindow.h" #include #include int main(int argc, char *argv[]) { QApplication a(argc, argv); //设置中文字体 防止乱码 a.setFont(QFont("Microsoft Yahei", 9)); //设置中文编码 #if (QT_VERSION <= QT_VERSION_CHECK(5,0,0)) #if _MSC_VER QTextCodec *codec = QTextCodec::codecForName("GBK"); #else QTextCodec *codec = QTextCodec::codecForName("UTF-8"); #endif QTextCodec::setCodecForLocale(codec); QTextCodec::setCodecForCStrings(codec); QTextCodec::setCodecForTr(codec); #else QTextCodec *codec = QTextCodec::codecForName("UTF-8"); QTextCodec::setCodecForLocale(codec); #endif MainWindow w; w.show(); return a.exec(); } 接下来重点来了,源文件CPP里为 #include "mainwindow.h" #include "ui_mainwindow.h" #include #include #include #include #include MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent) , ui(new Ui::MainWindow) { ui->setupUi(this); t = new QThread(); //QThread obj = new MyObj(); obj->moveToThread(t); qDebug()<<"main thread:"<<QThread::currentThread(); connect(t,SIGNAL(started()), obj, SLOT(doWork())); connect(obj,SIGNAL
提供的源码资源涵盖了Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值