QThread用法:启动与退出

QThread 有两种用法:

一,运用 QObject::moveToThread(QThread *targetThread)将功能类移入到线程QThread中

例子如下:

class Worker : public QObject 
{//创建普通的对象类
    Q_OBJECT

    public slots:
        void doWork(const QString &parameter) {
          QString result;
          /* ... here is the expensive or blocking operation ... */
          emit resultReady(result);
        }

    signals:
      void resultReady(const QString &result);
  };

  class Controller : public QObject
  {
      Q_OBJECT
      QThread workerThread;
  public:
        Controller(){
        Worker *worker = new Worker;
        worker->moveToThread(&workerThread);
        connect(&workerThread, &QThread::finished, worker, &QObject::deleteLater);//线程停止,释放资源
        connect(this, &Controller::operate, worker, &Worker::doWork);//线程实际工作代码
        connect(worker, &Worker::resultReady, this, &Controller::handleResults);
        workerThread.start();
        }
        ~Controller(){
            workerThread.quit();
            workerThread.wait();
      }
    public slots:
        void handleResults(const QString &);
    signals:
        void operate(const QString &);
  };

然后,worker 的槽函数的代码将在单独的线程中执行。但是,您可以自由地将worker 的槽函数连接到任何线程中来自任何对象的任何信号。通过一种名为queued connections的机制,在不同线程之间连接信号和插槽是安全的。

二 继承QThread,在run函数中实现子线程

class WorkerThread : public QThread
{
    Q_OBJECT
    void run() Q_DECL_OVERRIDE {
        QString result;
        /* ... here is the expensive or blocking operation ... */
        emit resultReady(result);

      exec();
    }
    signals:
        void resultReady(const QString &s);
};

void MyObject::startWorkInAThread()
{
    WorkerThread *workerThread = new WorkerThread(this);
    connect(workerThread, &WorkerThread::resultReady, this, &MyObject::handleResults);
    connect(workerThread, &WorkerThread::finished, workerThread, &QObject::deleteLater);
    workerThread->start();
}

在这个例子中,线程会在run函数运行结束后退出。只有调用了exec(),线程才会有event loop运行,意思是若没有调用exec(),exit(),quit()函数不会起作用。

必须记住,QThread实例位于实例化它的旧线程中,而不是调用run()的新线程中。这意味着QThread队列中的所有插槽都将在旧线程中执行。因此,希望在新线程中调用插槽的开发人员必须使用worker-object方法;不应该将新插槽直接实现到子类QThread中。

在子类化QThread时,请记住,构造函数在旧线程中执行,而run()在新线程中执行。如果从两个函数访问成员变量,则从两个不同的线程访问该变量。检查一下这样做是否安全。

一种不推荐的退出线程的用法:terminate()

  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
QThread退出可以通过调用quit()函数来实现。在代码中,可以使用thread->quit()来告诉线程的事件循环退出,返回代码为0(成功)。相当于调用QThread::exit(0)。如果线程没有事件循环,则此函数不执行任何操作。\[2\] 在示例代码中,可以看到在退出时使用了thread->quit()来告诉线程的事件循环退出,并且使用了thread->wait()来等待线程真正退出。这样可以确保线程在退出后再进行后续的操作。同时,还需要将thread和object设置为nullptr,以释放相应的资源。\[1\] 另外,还可以通过在Controller类的析构函数中调用workerThread.quit()来退出线程,并使用workerThread.wait()来等待线程真正退出。这样可以确保线程在析构时能够正常退出并释放资源。\[3\] #### 引用[.reference_title] - *1* *2* [Qt QThread安全退出](https://blog.csdn.net/qq_44365088/article/details/119087454)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [QThread用法启动退出](https://blog.csdn.net/fxbjye/article/details/83792922)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值