Qt设计子线程执行任务队列

  在子线程中维护一个任务队列,排队执行主线程中添加的耗时/后台任务。

  设计的任务类如下:

#ifndef TASK_H
#define TASK_H

#include <QObject>


// Demo是为了方便,将接口类和具体类写在了一起

// 任务接口类
class CTask
{
public:
    virtual ~CTask() {}
    virtual int doWork() = 0;
    virtual QString message() = 0;
};

// 结束任务,不做实质性操作,一般仅用作结束线程的标志
class CEndTask : public CTask
{
public:
    int doWork() override;
    QString message() override;
};


// 实际的任务实现类
class CMyTask : public CTask
{
public:
    CMyTask(int taskId);
    int doWork() override;
    QString message() override;

private:
    int id; // 任务id
};

#endif // TASK_H
#include "task.h"
#include <QThread>
#include <QDebug>
#include <QDateTime>

CMyTask::CMyTask(int taskId) : id(taskId)
{

}

int CMyTask::doWork()
{
    // 模拟耗时的任务,doWork里面不要执行任何操作GUI控件的操作
    for (int i = 0; i < 10; ++i) {
        QThread::msleep(1000);
        qDebug() << "[CMyTask:" << id << "]run in thread:" << QThread::currentThreadId() << " >>> "
                 << QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss zzz");
    }

    return 0;
}

QString CMyTask::message()
{
    return QString("MyTask");
}

int CEndTask::doWork()
{
    qDebug() << "[CEndTask Thread:" << QThread::currentThreadId() << "] >>> "
             << QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss zzz");
    return 0;
}

QString CEndTask::message()
{
    return QString("CEndTask");
}

  线程类:

#ifndef TASKTHREAD_H
#define TASKTHREAD_H

#include <QThread>
#include <QQueue>
#include <QWaitCondition>
#include <QMutex>

class CTask;
class CEndTask;

class TaskThread : public QThread
{
    Q_OBJECT
public:
    explicit TaskThread(QObject *parent = nullptr);
    ~TaskThread();

    void addTask(CTask *task); // 添加任务的接口
    void stopRun();            // 停止线程执行接口

protected:
    void run() override;

// 定义一些与主线程通信的信号
signals:
    void taskStart(const QString &message);
    void allTaskDone();

private:
    QQueue<CTask*> taskQueue; // 任务队列
    QWaitCondition taskAdded; // 任务等待条件
    QMutex mutex;             //
    CTask *EndTask;           // 结束任务
};

#endif // TASKTHREAD_H
#include <QMutexLocker>
#include <QDebug>
#include "task.h"
#include "taskthread.h"


TaskThread::TaskThread(QObject *parent): QThread(parent), EndTask(new CEndTask)
{
    start(); // 构造时即开启,也可以不立马开启线程
}

TaskThread::~TaskThread()
{
    {
        QMutexLocker locker(&mutex);
        // 清空任务队列
        while (!taskQueue.isEmpty()) {
            delete taskQueue.dequeue();
        }
    }

    if (this->isRunning()) {
        stopRun();// stop thread
    }

    wait();

    delete EndTask;
}


void TaskThread::addTask(CTask *task)
{
    // 必须加锁访问任务队列,防止并发访问发生冲突
    QMutexLocker locker(&mutex);
    taskQueue.enqueue(task); // 添加任务
    taskAdded.wakeOne();
}

void TaskThread::stopRun()
{
    QMutexLocker locker(&mutex);
    taskQueue.enqueue(this->EndTask); // 添加结束任务
    taskAdded.wakeOne();
}

void TaskThread::run()
{
    CTask *task = nullptr;
    qDebug() << "Task Thread ID:" << QThread::currentThreadId();

    forever {
        {
            QMutexLocker locker(&mutex);
            if (taskQueue.isEmpty()) {
                // 任务队列为空,等待有任务时唤醒
                taskAdded.wait(&mutex);
            }

            // 获取一个任务
            task = taskQueue.dequeue();

            // 判断是否是退出任务
            if (task == EndTask) {
                task->doWork(); // 执行结束任务
                break; // 跳出循环,结束线程
            }
        }

        if (task) {
            // 发送任务开始执行信号
            emit taskStart(task->message());

            // 执行任务
            task->doWork();

            // 执行完成将其删除
            delete task; task = nullptr;
        }

        // 检查任务是否全部完成
        {
            QMutexLocker locker(&mutex);
            if (taskQueue.isEmpty())
                qDebug() << "All Task Is Done Now!";
                emit allTaskDone();
        }
    }
    qDebug() << "Thread Run End!";
}

  使用Demo

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>

class TaskThread;

namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = nullptr);
    ~MainWindow();

private slots:
    void on_btnAddTask_clicked();
    void on_actionStop_triggered();

private:
    Ui::MainWindow *ui;
    TaskThread* thd = nullptr;
};

#endif // MAINWINDOW_H
#include "mainwindow.h"
#include <QDebug>
#include "ui_mainwindow.h"
#include "task.h"
#include "taskthread.h"

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);

    qDebug() << "Main Thread ID:" << QThread::currentThreadId();

    thd = new TaskThread(this);
    // 可以关联线程的信号
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::on_btnAddTask_clicked()
{
    static int id = 0;
    if (thd->isRunning())
        thd->addTask(new CMyTask(id++));
}


void MainWindow::on_actionStop_triggered()
{
    if (thd->isRunning())
        thd->stopRun();
}

测试效果如下:(点击3下任务添加按钮,添加3个任务,停止Action用于终止线程执行)

11:28:15: Starting D:\Desktop\CodePro\Qt\ThreadApp\Debug\debug\ThreadApp...
Main Thread ID: 0x35fc
Task Thread ID: 0x3b2c

[CMyTask: 0 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:26 763"
[CMyTask: 0 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:27 775"
[CMyTask: 0 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:28 777"
[CMyTask: 0 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:29 778"
[CMyTask: 0 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:30 779"
[CMyTask: 0 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:31 780"
[CMyTask: 0 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:32 781"
[CMyTask: 0 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:33 782"
[CMyTask: 0 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:34 783"
[CMyTask: 0 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:35 784"
[CMyTask: 1 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:36 785"
[CMyTask: 1 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:37 786"
[CMyTask: 1 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:38 787"
[CMyTask: 1 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:39 788"
[CMyTask: 1 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:40 789"
[CMyTask: 1 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:41 791"
[CMyTask: 1 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:42 792"
[CMyTask: 1 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:43 792"
[CMyTask: 1 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:44 793"
[CMyTask: 1 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:45 793"
[CMyTask: 2 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:46 794"
[CMyTask: 2 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:47 795"
[CMyTask: 2 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:48 796"
[CMyTask: 2 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:49 797"
[CMyTask: 2 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:50 799"
[CMyTask: 2 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:51 800"
[CMyTask: 2 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:52 801"
[CMyTask: 2 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:53 802"
[CMyTask: 2 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:54 803"
[CMyTask: 2 ]run in thread: 0x3b2c  >>>  "2021-05-11 11:28:55 803"
All Task Is Done Now!
[CEndTask Thread: 0x3b2c ] >>>  "2021-05-11 11:29:00 306"
Thread Run End!

11:29:07:D:/Desktop/CodePro/Qt/ThreadApp/Debug/debug/ThreadApp exited with code 0

 

posted @ 2021-05-11 11:37  碎银三二两  阅读(1308)  评论(0编辑  收藏  举报