解决Qt多线程中fromRawData函数生成的QByteArray数据不一致问题

解决Qt多线程中fromRawData函数生成的QByteArray数据不一致问题

icon

目录

    • 🔔 问题背景
    • 📄 问题代码
    • ❓ 问题描述
    • 🩺 问题分析
    • ✔ 解决方案


🔔 问题背景

在开发一个使用Qt框架的多线程应用程序时,我们遇到了一个棘手的问题:在不同线程中打印同一个QByteArray对象的内容时,得到了不一致的结果。这个问题出现在一个设备通信类中,该类使用一个工作线程来读取设备数据,然后通过信号-槽机制将数据传递到主线程进行处理。



📄 问题代码

代码结构如下:

class DeviceManager : public QObject {
public:DeviceManager() : QObject() {m_worker = new Worker();m_workerThread = new QThread(this);m_worker->moveToThread(m_workerThread);connect(m_workerThread, &QThread::started, m_worker, &Worker::readDevice);connect(m_worker, &Worker::dataReady, this, &DeviceManager::processData);m_workerThread->start();}~DeviceManager() {m_hidWork->breakFlag = false;m_hidWorkThread->quit();m_hidWorkThread->wait();m_hidWorkThread->deleteLater();}private:void processData(QByteArray data) {qDebug() << "Main thread: " << data.toHex(' ');}class Worker;Worker *m_worker;QThread *m_workerThread;
};class DeviceManager::Worker : public QObject {Q_OBJECT
public:Worker(QObject *parent = nullptr) : QObject(parent) {}bool breakFlag = true;void readDevice() {auto buffer = new char[1024];while(breakFlag) {size_t size = 5;memset(buffer, 0, size)// 这里是模拟读取设备数据buffer = {0x01, 0x02, 0x03, 0x04, 0x05};QByteArray deviceData = QByteArray::fromRawData(reinterpret_cast<char*>(buffer), size);qDebug() << "Worker thread: " << deviceData.toHex(' ');emit dataReady(QByteArray(deviceData));}}signals:void dataReady(QByteArray data);
};


❓ 问题描述

在运行这段代码时,当主线程进行了一个UI阻塞操作的时候,我们观察会到工作线程和主线程中打印的QByteArray内容不一致。例如:

Worker thread: 01 02 03 04 05
Main thread: 00 00 00 00 00

这显然不是我们期望的结果,因为两个线程应该打印相同的数据。



🩺 问题分析

经过仔细分析,我们发现问题的根源在于QByteArray::fromRawData()的使用方式。这个函数创建了一个共享底层数据QByteArray,而不是复制数据。这意味着:

  • fromRawData()创建的QByteArray与原始缓冲区共享数据
  • 当原始缓冲区被修改或释放时,QByteArray的内容可能变得无效
  • 在跨线程传递时,如果原始数据发生变化,接收线程可能会得到意外的结果


✔ 解决方案

解决这个问题的关键是确保在发送信号时创建QByteArray完整副本。以下是修改后的Worker::readDevice()方法:

void Worker::readDevice() {auto buffer = new char[1024];while(breakFlag) {size_t size = 5;memset(buffer, 0, size)// 这里是模拟读取设备数据buffer = {0x01, 0x02, 0x03, 0x04, 0x05};//QByteArray deviceData = QByteArray::fromRawData(reinterpret_cast<char*>(buffer), size);QByteArray deviceData((reinterpret_cast<char*>(buffer), size);  // 创建数据的副本qDebug() << "Worker thread: " << deviceData.toHex(' ');emit dataReady(deviceData);}	    
}

这里的改动看似很小,但却解决了问题:

  • 我们使用QByteArray(const char *data, int size)构造函数来创建QByteArray
  • 这个构造函数会复制提供的数据,而不是共享它。
  • 结果是一个独立的QByteArray对象,其内容不会受到原始缓冲区变化的影响。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/52403.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Redis的数据结构——Hash表

Redis使用Hash表作为其底层数据结构来存储键值对。每个Redis数据库都维护着两个哈希表(Hash Table):ht[0]和ht[1],其中ht[0]是主要的哈希表,ht[1]是在进行rehash操作时使用的临时表。 Hash表的结构 Redis的哈希表由一个数组和一个链表组成。数组中的每个元素是一个指针,…

数据赋能((185)——开发:提高数据价值密度——实施过程、应用特点

实施过程 提高数据价值密度的实施过程通常包括以下几个步骤&#xff1a; 数据收集&#xff1a;根据业务需求&#xff0c;收集相关的数据资源。数据清洗&#xff1a;对收集到的数据进行清洗和预处理&#xff0c;去除重复、错误和无关的信息。数据分析&#xff1a;运用统计方法…

windows系统删除指定端口的进程

1、打开终端 winr的方式&#xff0c;打卡运行框&#xff0c;输入cmd按下回车&#xff0c;即可打开终端。 2、查询端口进程 netstat -ano | findstr 8080 3、根据进程ID删除进程 taskkill -PID 9340 -F 感谢您的阅读&#xff0c;欢迎参观我的个人网站&#xff1a;小嗨词典…

使用 LangGraph 构建工作流, 实现与虚拟女友对话

文章目录 简介背景流程图代码实现 简介 介绍了如何使用 LangGraph 搭建一个基于聊天机器人的工作流&#xff0c;具体实现了一个虚拟女友的角色扮演游戏。 通过流程图展示了构建完成的状态图&#xff0c;并介绍了各个节点的功能&#xff0c;如接收用户输入、生成对话等。提供了…

面向对象程序设计原则——迪米特法则

迪米特法则 迪米特法则&#xff08;Law of Demeter, LoD&#xff09;迪米特法则设计原则迪米特法则的优势应用场景实践示例迪米特法则的体现分析总结 迪米特法则&#xff08;Law of Demeter, LoD&#xff09; 迪米特法则&#xff0c;也称为最少知识原则&#xff08;Least Know…

如何使用ssm实现保险业务管理系统设计与实现

TOC ssm131保险业务管理系统设计与实现jsp 绪论 1.1 研究背景 当前社会各行业领域竞争压力非常大&#xff0c;随着当前时代的信息化&#xff0c;科学化发展&#xff0c;让社会各行业领域都争相使用新的信息技术&#xff0c;对行业内的各种相关数据进行科学化&#xff0c;规…

ArcGIS Pro基础:设置2个窗口同步联动界面

如上所示&#xff0c;通过1步骤&#xff0c;新建了2个地图窗口&#xff0c;得到2和3所表示的【地图1】、【地图2】&#xff0c;一个是影像图&#xff0c;另一个是地形图&#xff0c; 假如有个需求&#xff0c;是将2个窗口联动起来&#xff1a;在观察影像的同时&#xff0c;也同…

多人协作开发git merge合并功能出现冲突时解决思路

目录 问题背景描述 解决思路 解决详细步骤 经验适用场景 问题背景描述 现在有一个本地分支A&#xff0c;远端其中两个分支B、C&#xff0c;其中C为主分支&#xff0c;A和B目前版本是一致的&#xff0c;且都比较新&#xff0c;C的版本比较落后&#xff0c;现在需要从分支B合…

[000-01-022].第06节:RabbitMQ中的交换机介绍

1.什么是Exchanges(交换机&#xff09;: 1.RabbitMQ 消息传递模型的核心思想是: 生产者生产的消息从不会直接发送到队列。实际上&#xff0c;通常生产者甚至都不知道这些消息传递传递到了哪些队列中2.生产者只能将消息发送到交换机(exchange)&#xff0c;交换机工作的内容非常…

Android Room DataBase

Room数据库是在Sqlite的基础上&#xff0c;进行了封装和优化。这让我们可以摆脱&#xff0c;繁琐的数据库操作 在module的gradle里面&#xff0c;加入: dependencies {annotationProcessor "androidx.room:room-compiler:2.3.0"implementation androidx.room:room-…

调用股票网站接口读取大A数据——个股资金流入趋势

以某股票为例&#xff0c;调用自定义的一个类&#xff0c;读取数据。 class BigAData:# 获取资金流向数据def get_money_flow(self, stock_code, page1, num20, sortopendate, asc0):该函数通过股票代码从新浪财经API获取资金流向数据。参数包括股票代码、页数、每页数量、排序…

CSS3【待总结学习】

CSS3是Cascading Style Sheets&#xff08;层叠样式表&#xff09;的第三个版本&#xff0c;它是前端开发中用于控制网页布局和样式的重要技术。CSS3在CSS2的基础上引入了众多新特性和功能&#xff0c;大大增强了网页设计和交互的能力。以下是对CSS3的详细解析&#xff1a; 一…

jenkins最佳实践(一):jenkins安装与部署

各位小伙伴们大家好呀&#xff0c;我是小金&#xff0c;下面我将记录学习jenkins的系列文章与心得&#xff0c;一方面用于博主的自我记录&#xff0c;一方面如果能帮助到正在浏览这篇文章的小伙伴&#xff0c;那更好不过了&#xff0c;本篇文章主要讲述jenkins的安装以及安装je…

Redis篇一:初识Redis

文章目录 前言1. 初始Redis2. MySQL VS Redis3. 什么是分布式系统&#xff08;也是一种处理大量数据时的处理方式&#xff09;3.1 单机架构3.2 数据库与应用服务分离3.3 负载均衡3.4 数据库读写分离3.5 引入缓存&#xff08;Redis&#xff09;3.6 数据库分库分表3.7 引入微服务…

计算机毕业设计选题推荐-OA办公管理系统-Java/Python项目实战

✨作者主页&#xff1a;IT毕设梦工厂✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Py…

白酒与青年文化:潮流与传统的碰撞

在时代的洪流中&#xff0c;青年文化如同一股涌动的潮流&#xff0c;不断冲击着传统的边界。而白酒&#xff0c;作为中国传统文化的瑰宝&#xff0c;也在这一潮流中找到了新的表达方式。今天&#xff0c;我们就来探讨一下白酒与青年文化之间的碰撞与整合&#xff0c;以及豪迈白…

一文带你弄清楚基站是什么

我们每天都在通过手机、电脑等设备拨打电话和传递消息。然而&#xff0c;你是否曾深思过&#xff0c;这些来电显示和信息内容究竟是如何跨越距离&#xff0c;准确无误地从一个人传递到我们手中的呢&#xff1f;或许&#xff0c;有些细心的人已经留意到&#xff0c;在手机屏幕的…

【Vue3】编程式路由导航

【Vue3】编程式路由导航 背景简介开发环境开发步骤及源码总结 背景 随着年龄的增长&#xff0c;很多曾经烂熟于心的技术原理已被岁月摩擦得愈发模糊起来&#xff0c;技术出身的人总是很难放下一些执念&#xff0c;遂将这些知识整理成文&#xff0c;以纪念曾经努力学习奋斗的日…

从匿名内部类到Lambda表达式:Java编程的优雅进化

匿名内部类 首先我们先来介绍一下什么是匿名内部类 匿名内部类&#xff1a;java中一种特殊的类定义方式&#xff0c;它允许你在需要实现一个接口或继承一个类的地方直接定义一个该接口或类的匿名子类。若想创建一个派生类的对象&#xff0c;并且对象只创建一次&#xff0c;可…

微服务事务管理

1.分布式事务问题 1.1.本地事务 本地事务&#xff0c;也就是传统的单机事务&#xff0c;在传统数据库事务中&#xff0c;必须要满⾜四个原则&#xff1a; 1.2.分布式事务 分布式事务&#xff0c;就是指不是在单个服务或单个数据库架构下&#xff0c;产⽣的事务&#xff0c;例…