QT moveToThread线程理解

一、moveToThread创建开启线程步骤:

(1)创建继承自QObject类,实现槽函数。

(2)将QObject类通过moveToThread方法移到QThread线程中,使QObject类依附于线程。

(3)连接信号槽,槽必须是QObject类中函数。

(4)QThread调用start开启线程。

二、程序代码:

#ifndef MYOBJECT_H
#define MYOBJECT_H
 
#include 
#include 
class MyObject : public QObject
{
    Q_OBJECT
public:
    explicit MyObject(QObject *parent = nullptr);
    ~MyObject();
 
    QTimer *timer;
 
public slots:
    void slotStart();
 
 
signals:
 
};
 
#endif // MYOBJECT_H
#include "myobject.h"
#include 
#include 
#include 
MyObject::MyObject(QObject *parent) : QObject(parent)
{
 
}
 
MyObject::~MyObject()
{
    qDebug()<<"执行了MyObject的析构";
}
 
void MyObject::slotStart()
{
    qDebug()<<"子线程 id"<start(2000);   
}
#ifndef MAINWINDOW_H
#define MAINWINDOW_H
 
#include 
#include "myobject.h"
 
QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACE
 
class MainWindow : public QMainWindow
{
    Q_OBJECT
 
public:
    MainWindow(QWidget *parent = nullptr);
    ~MainWindow();
 
    QThread *thread;
    MyObject *object;
 
private slots:
    void on_pushButton_clicked();
 
    void on_pushButton_2_clicked();
 
signals:
    void signalStart();
 
private:
    Ui::MainWindow *ui;
};
#endif // MAINWINDOW_H
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include 
#include 
#include 
 
MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
{
    ui->setupUi(this);
}
 
MainWindow::~MainWindow()
{
    delete ui;
}
 
void MainWindow::on_pushButton_clicked()
{
    qDebug()<<"点击了开始,主线程id"<moveToThread(thread);
    connect(this,&MainWindow::signalStart,object,&MyObject::slotStart);
    connect(thread,&QThread::finished,object,&QObject::deleteLater);
    thread->start();
 
    emit signalStart();
}
 
void MainWindow::on_pushButton_2_clicked()
{
    qDebug()<<"点击了停止,线程停止了"<quit();
    thread->wait();
    qDebug()<<"释放了thread";
    delete thread;
    thread = NULL;
}

ui界面

QT moveToThread线程理解_第1张图片

 输出结果

QT moveToThread线程理解_第2张图片

 三、补充:

1. 如果线程已经start运行,重复调用start不会进行任何处理。
2. 槽函数只在一个线程中,如果当前槽函数未执行完,下一次槽函数又要进来,会阻塞槽函数的执行。
3. 停止线程时,如果线程中槽函数未执行完,不会立刻停止线程,会等待槽函数执行完成之后停止线程。
4. 关闭线程一般先quit,然后在wait,原因是在执行quit()后,调用wait()来等待QThread子线程的结束,这样就能保证在清除QThread时,其子线程是停止运行的。
5. 使用线程的finished信号,QObject的deleteLater槽函数,会在线程quit后,发送finished信号,释放QObject类。

通过程序测试得出以下结论:
1.如果是在QObject类中即WriteThread类中创建的QTimer,不管加没加this,触发的槽函数均依附于子线程。
2.如果在QObject类中即WriteThread类中创建的其他类,加this,该类依附于子线程;不加this,该类依附于主线程。
3.如果是MainWindow中创建的其他类传到QObject类中即WriteThread类中,不管有没有加this,该类均依附于主线程。
4.信号所在线程看的是信号发出的时机是在子线程中还是主线程中,而不是信号发出者所依附的线程。
5.当信号在子线程中发出,槽函数在子线程中接收,默认连接为直接连接;
  当信号在主线程中发出,槽函数在子线程中接收,默认连接为队列连接;
  当信号在子线程中发出,槽函数在主线程中接收,默认连接为队列连接;
  当信号在主线程中发出,槽函数在主线程中接收,默认连接为直接连接;
即信号发出时的所在线程和信号接收者所属线程相同时,属于直接连接;不同时,属于队列连接

以上均为自己的理解,如果有不正确的地方,欢迎指出。

你可能感兴趣的:(QT,qt,开发语言)