在C ++中使用QThread多线程编程 继承QThread 处理线程上的信号

问题描述

我想学习一些有关多线程的知识,因为我的本科论文中需要它。我认为我在理论上不太了解,但在实践上却不了解。

首先,我尝试创建一个非常简单的程序,在这里我想学习如何使用QThread。我的小程序会测试目录中的所有文件是否为.txt文件。

有一个函数可以测试一个文件是否包含.txt及其在下一个函数中使用的功能。 我知道我知道这不是测试文件是否为.txt的完美测试,但这确实无关紧要。这是关于别的事情。

void FileTester::testIfTxt(QString &file){
    if( file.contains(".txt"))
        std::cout << file.toStdString() << " is .txt file" << std::endl;
    else
        std::cout << file.toStdString() << " is NOT .txt file" << std::endl;
}

这里是不使用多线程进行测试的功能。

void FileTester::normalTest(){
    QDir folderPath("D:\\programming\\C++\\multithreading-qt\\files");
    QStringList filesList = folderPath.entryList(QDir::NoDotAndDotDot | QDir::Files);

    auto t_start = std::chrono::high_resolution_clock::now();

    for(QString& file : filesList){
        testIfTxt(file);
    }

    auto t_end = std::chrono::high_resolution_clock::now();
    double elapsed_time_ms = std::chrono::duration<double,std::milli>(t_end-t_start).count();

    std::cout << elapsed_time_ms << " ms" << std::endl;
}

现在我想做同样的事情,但是要使用多线程。首先,我们可以使用3个线程。第一个线程将获取第一个文件并进行测试,而第二个线程将获取第二个文件并进行测试,而第三个线程将获取第三个文件并进行测试。当第一个线程结束工作时,则将nex未经测试的文件和测试等..

我从那里的函数开始,但是我真的不明白如何使用这样的东西。我从这样的事情开始。

void FileTester::multiThreadTest(){
    auto t_start = std::chrono::high_resolution_clock::now();

    allMultiThreadTest();

    auto t_end = std::chrono::high_resolution_clock::now();
    double elapsed_time_ms = std::chrono::duration<double,std::milli>(t_end-t_start).count();

    std::cout << elapsed_time_ms << " ms" << std::endl;
}


void FileTester::allMultiThreadTest(){
    QThread* thread1 = new QThread();
    QThread* thread2 = new QThread();
    QThread* thread3 = new QThread();

    thread1->start();
    thread2->start();
    thread3->start();

    for(QString& file : filesList){
        testIfTxt(file);
    }
}

同样,这不是我的学士学位论文中的代码。我只想学习如何在这个简单的程序上使用多线程。谢谢您的帮助。

解决方法

QThread documentation与大多数Qt文档一样,非常好,但是代码示例可能有点让人不知所措。

基本上有两种使用QThread的方式:

继承QThread

在要在线程上运行某些代码时有用,在代码完成后退出线程。

创建QThread的子类并覆盖其run()函数。您在该函数中放入的代码将在线程中运行,当函数返回时,该线程退出。

要启动线程,只需创建子类的实例并在其上调用start()

处理线程上的信号

在希望将线程保留在后台并偶尔发送一些工作时有用。这种方法需要一些样板代码,因为您需要一个对象来发射信号,而另一个对象要通过插槽接收信号。

创建QObject的子类,并为其选择一个公共位置。

创建QThread的实例(此处无需子类)。 run()的默认实现包含一个事件循环,该事件循环仅等待信号并运行相应的插槽,就像QApplication::exec()对您的主线程一样。

创建QObject子类的实例。使用QObject::moveToThread()将后者移至新线程。这样的效果是,信号调用的所有插槽都将在该线程上运行,而不是在主线程上运行。参见thread affinityqueued connections

将插槽连接到主线程上任何QObject子类发出的信号。

要将工作发送到线程,只需发出该信号即可。

注意:Qt的元类型系统必须知道传递给信号的任何参数。大多数内置的Qt类型都可以。如果您违反此规则,则会在运行时收到警告

相关问答

依赖报错 idea导入项目后依赖报错,解决方案:https://blog....
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下...
错误1:gradle项目控制台输出为乱码 # 解决方案:https://bl...
错误还原:在查询的过程中,传入的workType为0时,该条件不起...
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct...