Qt多线程数据处理及分析

一、线程

        线程(英语:thread)是操作系统能够进行运算调度的最小单位。它被包含在进程之中,是进程中的实际运作单位。一条线程指的是进程中一个单一顺序的控制流,一个进程中可以并发多个线程,每条线程并行执行不同的任务。

二、Qt多线程

使用QThread类。
作用:用线程来处理那些耗时的后台操作,从而让主界面能及时响应用户的请求操作。

运行方式

QThread的启动由start()开始

·start() 函数会在被触发时开启线程,一般放在线程初始化或者被SIGNAL触发时启动。

QThread 的执行从 run() 函数的执行开始。

·run() 函数通过调用 exec() 函数来启动事件循环机制,并且在线程内部处理 Qt 的事件。

QThread 结束可以使用quit()函数。

三、Qt 互斥锁

        线程互斥是指在多线程并发执行时,为避免多个线程访问共享资源时发生冲突而采取的一种机制。

导致问题产生的原因和解决方法

        如果多个线程同时访问同一共享资源,可能会导致数据不一致、资源竞争和死锁等问题。为了避免这些问题,可以使用互斥锁(Mutex)来保护共享资源。互斥锁是一种同步机制,用于控制多个线程对共享资源的访问。当一个线程获得了互斥锁,其他线程就无法获得该锁,直到该线程释放互斥锁为止。

四、多线程接收数据

        在进行基于通信对软件进行开发时,往往会出现数据从不同的端口传过来,且数据量可能很大,此时光靠主线程没有办法做到实时处理数据,此时可以使用线程进行异步处理。

        接下来上代码,模拟两个数据(造数据的线程)送到一个接收线程中。

createdata1_thread.h

#ifndef CREATEDATA1_THREAD_H
#define CREATEDATA1_THREAD_H

#include <QThread>
#include "handledata_thread.h"
#include <QDebug>

class CreateData1_Thread:public QThread
{
public:
    CreateData1_Thread(HandleData_Thread *m_pThread);
    HandleData_Thread *m_pSend;
    QString CreateData(QString str);
    bool SendData(QString str);
    ~CreateData1_Thread();
private:
    void run();
};

#endif // CREATEDATA1_THREAD_H

createdata1_thread.cpp

#include "createdata1_thread.h"
#include "mainthread.h"
CreateData1_Thread::CreateData1_Thread(HandleData_Thread *m_pThread)
{
    start();
    m_pSend=m_pThread;

}
CreateData1_Thread::~CreateData1_Thread()
{

}

QString CreateData1_Thread::CreateData(QString str)
{
    str="a";
    return str;
}

bool CreateData1_Thread::SendData(QString str)
{
    m_pSend->ReceiverData(str);
    return 1;
}

void CreateData1_Thread::run()
{
    QString str;
    str=CreateData(str);
    while(1)
    {
        SendData(str);
        QThread::sleep(1);
    }
    qDebug()<<"1runend!";
}

由于两个造数据的代码基本相同,唯一不同的是传输的字符不同。

接下来是接收数据的线程

handledata_thread.h

#ifndef HANDLEDATA_THREAD_H
#define HANDLEDATA_THREAD_H

#include <QThread>
#include <QDebug>
#include <QMutex>

class HandleData_Thread:public QThread
{
public:
    HandleData_Thread();
    ~HandleData_Thread();
    bool printData();

public slots:
    bool ReceiverData(QString str);

private:
    QStringList strlist;
    QMutex datalock;
    void run();
};

#endif // HANDLEDATA_THREAD_H

handledata_thread.cpp

#include "handledata_thread.h"

HandleData_Thread::HandleData_Thread()
{
    start();
}


HandleData_Thread::~HandleData_Thread()
{

}

bool HandleData_Thread::ReceiverData(QString str)
{
    datalock.lock();
    strlist.append(str);
    datalock.unlock();
    return 1;
}

bool HandleData_Thread::printData()
{
    QStringList p_str;
    datalock.lock();
    p_str=strlist;
    datalock.unlock();
    qDebug()<<p_str;
    return 1;
}

void HandleData_Thread::run()
{
    while(1)
    {
        printData();
        QThread::sleep(3);
    }
    qDebug()<<"Hrunend!";
}

注意在接收数据的时候一定要在使用共享资源的时候加上互斥锁,避免因为冲突导致程序崩溃。由于qt输出框对于打印数据的限制,且本程序适用于学习和测试多线程各项操作故使用了QThread::sleep();

datalock.lock();
strlist.append(str);
datalock.unlock();
--------------------
datalock.lock();
p_str=strlist;
datalock.unlock();

main.h

#ifndef MAINTHREAD_H
#define MAINTHREAD_H

#include <QObject>
#include "createdata1_thread.h"
#include "createdata2_thread.h"
#include "handledata_thread.h"
#include <QDebug>
class mainThread : public QObject
{
    Q_OBJECT
public:
    explicit mainThread(QObject *parent = nullptr);
    CreateData1_Thread *m_C1Thread;
    CreateData2_Thread *m_C2Thread;
    HandleData_Thread *m_HThread;
 
signals:

public slots:
};

#endif // MAINTHREAD_H

main.cpp

#include "mainthread.h"

mainThread::mainThread(QObject *parent) : QObject(parent)
{

}

int main()
{
    mainThread *m_pThread;
    m_pThread=new mainThread();
    m_pThread->m_HThread=new HandleData_Thread();
    m_pThread->m_C1Thread=new CreateData1_Thread(m_pThread->m_HThread);
    m_pThread->m_C2Thread=new CreateData2_Thread(m_pThread->m_HThread);
    while(1)
    {

    }
    return 1;
}

        在主线程中,需要注意的是我们要对线程进行初始化,一般要在线程的初始化函数中写start();,确保线程一直处于待启动或启动状态,如果需要一直启动线程,就需要保证启动线程后,线程一直处在while(1)的循环下不结束,一个start()对应一次run();所以在使用线程是要确保你进行操作的线程是你一直使用的线程,而不是通过不断初始化或者start()的线程,如果不断初始化或者不断start()可能会因为内存不够或者内存冲突导致程序崩溃。

        最后是输出的结果:

()
("a", "b", "a", "b", "a", "b")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b")
("a", "b", "a", "b", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "a", "b", "b", "a", "b", "a", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a", "a", "b", "b", "a")

可以看出接收线程的贡献数据并不是被轮流使用的,而是根据cpu对时间片进行判定随机分配的。

  • 21
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
Qt中,多线程处理大量数据可以通过使用过滤器、映射器和简化器的概念来实现。这种方法通过依次处理数据并创建少量的辅助线程来提高处理效率,从而减少系统开销。另外,Qt还提供了多种方式来分配并执行处理数据的线程。 一种方式是利用QtConcurrent::run()函数,在Qt全局线程池的辅助线程中运行函数来处理数据。这种方法可以将处理函数放入一个线程中执行,让Qt自动管理线程的创建和销毁。 另一种方式是创建QRunnable对象并在Qt全局线程池辅助线程中处理数据。通过创建QRunnable对象,可以将要处理的数据分配给不同的线程,并由Qt管理线程的执行。 还有一种方式是创建QThread对象并将其作为辅助线程来处理数据。这种方式需要手动管理线程的创建和销毁,但可以更灵活地控制线程的执行。 无论是哪种方式,多线程处理大量数据都需要注意死锁的风险,并且编写拥有多线程处理功能的程序相对于单线程来说更加困难,需要谨慎对待。 综上所述,Qt提供了多种方法和类来实现多线程处理大量数据,包括过滤器、映射器和简化器的概念,以及QtConcurrent::run()、QRunnable和QThread等类的使用。根据具体的需求,可以选择合适的方式来实现多线程处理大量数据的功能。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [Qt高级编程之多线程处理](https://blog.csdn.net/weixin_38880029/article/details/129482609)[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^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

傻瓜小白

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值