正确实现 QThread 的方法(附示例)

这篇技术博客探讨了在 Qt 框架中正确实现 QThread 类的方法。传统 Qt 文档建议继承 QThread 类并重写其 run 方法,这虽然在某些情况下可行,但并不是推荐的最佳实践。以下内容将探讨这种方法的利弊,并提供一个更好的实现 QThread 的示例。

原始示例:继承 QThread 并重写 run 方法

在 Qt 4.7 的 QThread 文档中,推荐下面的实现方式:

cpp 复制代码
class MyThread : public QThread
{
public:
    void run();
};

void MyThread::run()
{
    QTcpSocket socket;
    // 连接 QTcpSocket 的信号到一些有意义的槽
    ...
    socket.connectToHost(hostName, portNumber);
    exec();
}

在这种方法中,我们通过重写 run 方法来启动新的线程,并在该线程中运行事件循环。然而,这种方式存在一些问题,尤其是当在多线程中操作对象时,可能会引发线程安全问题。

推荐方法:使用 QObject 及其信号槽机制

一种更推荐的方法是将实际的工作放在 QObject 的子类中,并使用 QThread 来运行该对象。这种方法更符合面向对象编程的原则,并且更安全。以下是一个完整的示例:

工作器类 (Worker)

cpp 复制代码
class Worker : public QObject
{
    Q_OBJECT
public slots:
    void doWork(const QString &parameter) {
        // 执行具体的工作
        // ...
        emit resultReady(result);
    }

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

在这个示例中,我们定义了一个 Worker 类,它继承自 QObject 并包含一个用于执行工作的槽和一个用于发送结果的信号。

控制器类 (Controller)

cpp 复制代码
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 &result) {
        // 处理工作结果
    }

signals:
    void operate(const QString &parameter);
};

Controller 类中,我们创建了一个 Worker 实例并将其移到新线程中,并通过信号和槽机制连接各个事件和处理函数。这种方法避免了直接继承 QThread,使得代码更清晰易懂,同时也更加安全和稳定。

结论

尽管继承 QThread 并重写其 run 方法是 Qt 文档中介绍的一种实现线程的方式,但为了更好的线程管理和代码维护,更推荐使用上面介绍的使用 QObject 和信号槽机制的方法。这不仅符合面向对象编程的原则,还能避免潜在的线程安全问题,提升代码的稳定性和可读性。

相关推荐
卷卷的小趴菜学编程24 分钟前
c++之多态
c语言·开发语言·c++·面试·visual studio code
OopspoO1 小时前
C++ 标准库——函数对象和函数适配器
c++
kyle~1 小时前
thread---基本使用和常见错误
开发语言·c++·算法
攻城狮7号1 小时前
【第三节】C++设计模式(创建型模式)-单例模式
c++·单例模式·设计模式
oioihoii2 小时前
深入理解 C++17 的缓存行接口
java·c++·缓存
大白的编程日记.2 小时前
【C++笔记】C+11深度剖析(三)
c语言·开发语言·c++
老菜鸡mou2 小时前
[OD E 100] 生成哈夫曼树
数据结构·c++
别NULL3 小时前
机试题——编辑器
c++·算法
和光同尘@3 小时前
56. 合并区间 (LeetCode 热题 100)
c语言·开发语言·数据结构·c++·算法·leetcode·职场和发展
zh路西法3 小时前
【C++委托与事件】函数指针,回调机制,事件式编程与松耦合的设计模式(上)
开发语言·c++·观察者模式·设计模式