当前位置: 首页 > news >正文

QT将QBytearray的data()指针赋值给结构体指针变量后数据不正确的问题

1、问题代码

#include <QCoreApplication>#pragma pack(push, 1)
typedef struct
{int a;		// 4字节float b;	// 4字节char c;		// 1字节int *d;		// 8字节
}testStruct;
#pragma pack(pop)#include <QByteArray>
#include <QDebug>int main()
{testStruct structA;structA.a = 1;structA.b = 2;structA.c = 'a';structA.d = new int[10];for (int i = 0; i < 10; ++i) {structA.d[i] = i;}QByteArray arr;arr.resize(49);memcpy(arr.data(), &structA, 9);memcpy(arr.data() + 9, structA.d, 40);testStruct *sb = reinterpret_cast<testStruct *>(arr.data());sb->d = reinterpret_cast<int *>(arr.data() + 9);for (int i = 0; i < 10; ++i) {qDebug() << sb->d[i];}return 0;
}

输出结果

40642209
0
2
3
4
5
6
7
8
9

2、调试过程1

经过调试指针转换后,发现本来设置的1字节对其变成了4字节对齐
QBytearray的data()指针被转换后会按照4字节自动对齐填充,不知为何会这样
在这里插入图片描述>在这里插入图片描述

3、调试过程2

当我采用系统默认的8字节对齐后,又发现无法正确的对结构体内指针进行赋值
下面这行代码,按理来说指针赋值,sb->d应该指向arr.data() + 16的地址上

typedef struct
{int a;		// 4字节float b;	// 4字节char c;		// 8字节int *d;		// 8字节
}testStruct;
sb->d = reinterpret_cast<int *>(arr.data() + 16);;

但实际上出现了一个奇怪的现象,在arr.data() + 16 的内存起始地址上,莫名填充了8个字节,而这8个字节恰好就是自己的内存地址??????
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
其中00 00 00 00 00 f6 43 48 等于 16139080,输出的第一个值正好为自己的地址
由此得出,结构体指针的赋值会将值和地址一并一起赋值?匪夷所思

4、测试代码

测试一下,对于原始的char* c指针是否也有这样的情况

#include <QCoreApplication>#pragma pack(push, 1)
typedef struct
{int a;float b;char c;int *d;
}testStruct;
#pragma pack(pop)#include <QByteArray>
#include <QDebug>int main()
{testStruct structA;structA.a = 1;structA.b = 2;structA.c = 'a';structA.d = new int[10];for (int i = 0; i < 10; ++i) {structA.d[i] = i;}QByteArray arr;arr.resize(49);memcpy(arr.data(), &structA, 9);memcpy(arr.data() + 9, structA.d, 40);testStruct *sb = reinterpret_cast<testStruct*>(arr.data());// 分配一片char数组,假设我们需要存储3个floatchar *charBuffer = new char[10 * sizeof(int)];// 假设填充一些数据int exampleData[10] = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9};memcpy(charBuffer, exampleData, 10 * sizeof(int));sb->d = reinterpret_cast<int*>(charBuffer);// (*sb).d = reinterpret_cast<int*>(arr.data() + 16);for (int i = 0; i < 10; ++i) {qDebug() << sb->d[i];}return 0;
}

输出正常,给我整得一头雾水

5、问题解析

最后思来想去,终于发现了问题所在
结构体指针sb->d和需要解析的数据块在同一块内存地址,地址空间重合了!!!

在给sb->d的值赋值时,其实就是修改sb->d所在内存空间的8个字节为新的地址0x12345678,但是这块地址又和自己的数据地址重合了,所以导致了数据地址的前8个字节被覆盖

http://www.lryc.cn/news/455805.html

相关文章:

  • 修改银河麒麟操作系统V10(SP1)网卡名称为ethx
  • MySQL多表查询:标量子查询
  • C++学习笔记----8、掌握类与对象(六)---- 操作符重载(1)
  • Ascend C 自定义算子开发:高效的算子实现
  • 面向对象技术——设计模式
  • 2024 Mysql基础与进阶操作系列之MySQL触发器详解(20)作者——LJS[你个小黑子这都还学不会嘛?你是真爱粉嘛?真是的 ~;以后请别侮辱我家鸽鸽]
  • 找不到concrt140.dll如何修复,快来试试这6种解决方法
  • 年会工作会议会务报名签到小程序开源版开发
  • UE C++ 实时加载模型的总结
  • 实施威胁暴露管理、降低网络风险暴露的最佳实践
  • 51.哀家要长脑子了!
  • Overleaf 无法显示图片
  • 如何实现 C/C++ 与 Python 的通信?
  • 音视频入门基础:FLV专题(13)——FFmpeg源码中,解析任意Type值的SCRIPTDATAVALUE类型的实现
  • jvm里的metaspace oom 排查问题思路-使用MAT
  • 2025舜宇招聘【内推码】
  • APP自动化搭建与应用
  • kafka-windows集群部署
  • 4个顶级的大模型推理引擎
  • Oracle中ADD_MONTHS()函数详解
  • 【SQL】掌握SQL查询技巧:高效数据整合与查询优化
  • 一个月学会Java 第5天 控制结构
  • 世界职业院校技能大赛(大数据技术与应用)参赛项目介绍内容模拟示例参考
  • 【Python】文件及目录
  • OpenHarmony(鸿蒙南向开发)——标准系统方案之瑞芯微RK3566移植案例(下)
  • 霓虹灯数字时钟(可复制源代码)
  • 大模型微调技术之 LoRA:开启高效微调新时代
  • 【Vue】Vue2(2)
  • 如何实现一个基于 HTML+CSS+JS 的任务进度条
  • 学会流体力学,冬天洗澡再也不冷啦