如何将 char*强制转换为结构体,此结构体是混合类型的,有long,有char

raywangpeiqin 2012-03-31 01:06:13
结构体类型为此:
typedef struct _Rec
{
long origin_id;
char rerch_id[5];
char fermno_id[6];
char lard_id[7];
char trade_type;
char App_id[10];
long Sys_exp;
long Cup_exp;
} Rec;
字符串为char*ss = "0001aaaaabbbbbbcccccccmssssssssss00030006";
...全文
1399 10 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
10 条回复
切换为时间正序
请发表友善的回复…
发表回复
cobra_chen 2012-04-02
  • 打赏
  • 举报
回复
原因有2个去掉
[Quote=引用 9 楼 的回复:]

确实有内存不对齐的问题。
这也是因为windows分配内存会自动补位。
-
你可以实际代码测试一下,强制转换的效果。
楼主这样强制转换,本来就是不安全的。
不安全的原因:
1.不跨环境
2.要对内存存储了解。
char arr[4],强制转换为long(32位的环境long占4byte)这样的数据会错误。表面看上去,楼主这样强制转换出来之后,origin_id应该为1,实际上是一……
[/Quote]
cobra_chen 2012-04-02
  • 打赏
  • 举报
回复
确实有内存不对齐的问题。
这也是因为windows分配内存会自动补位。
-
你可以实际代码测试一下,强制转换的效果。
楼主这样强制转换,本来就是不安全的。
不安全的原因:
1.不跨环境
2.要对内存存储了解。
char arr[4],强制转换为long(32位的环境long占4byte)这样的数据会错误。表面看上去,楼主这样强制转换出来之后,origin_id应该为1,实际上是一个很大的数字。
原因有2个
头部的四个数据可以理解成char arr[4] = {'0','0','0','1'},'0'的ASCII值为48。'1'为49。因为内存是反序存储的,所以内存中表达方式为:字符串形式{'1','0','0','0'},十六进制:0x31 0x30 0x30 0x30,二进制:00110001,00110000,00110000,00110000,十进制:825241648。
如果要保证char arr[4]正确的强制转换为long,1,可以通过自己写规则转换,2,按照内存存储机制转换。如果想要转换为正确的数据。可以写成:

测试OK,环境vs2005。
下面是改进后的结构体,可以用sizeof(Rec)比较两个结构体的区别。

#pragma pack(push)
#pragma pack(1)
typedef struct _Rec
{
long origin_id;
char rerch_id[5];
char fermno_id[6];
char lard_id[7];
char trade_type;
char App_id[10];
long Sys_exp;
long Cup_exp;
} Rec;
#pragma pack(pop)
Rec rec = {0};
char *ss = "0001aaaaabbbbbbcccccccmssssssssss00030006";

rec = *(Rec *)ss;
memcpy(&rec ,ss ,sizeof(Rec));

char *des_ss = "\1\0\0\0aaaaabbbbbbcccccccmssssssssss\3\0\0\0\6\0\0\0";
rec = *(Rec *)des_ss;
memcpy(&rec ,des_ss ,sizeof(Rec));

[Quote=引用 7 楼 的回复:]

请楼上的各位大神 别误导人好么,你仔细看眼 struct _Rec 明显内存不对齐,内部有大量的空位,你memcpy 能 copy 我就给跪了

3楼正解 你只能写一个解析函数 然后 再手动赋值给 struct _Rec
[/Quote]
raywangpeiqin 2012-03-31
  • 打赏
  • 举报
回复
我自己是觉得要一个一个解析的,但是有一个很不好说话的人跟我说这个是可以强制类型转换的,我想可能是自己太菜了,以为可以的,看来只能一个一个解析了
likejieicy 2012-03-31
  • 打赏
  • 举报
回复
请楼上的各位大神 别误导人好么,你仔细看眼 struct _Rec 明显内存不对齐,内部有大量的空位,你memcpy 能 copy 我就给跪了

3楼正解 你只能写一个解析函数 然后 再手动赋值给 struct _Rec
酱油党 2012-03-31
  • 打赏
  • 举报
回复
感觉 结构体内部成员变量在内存中的分布机制不确定,(如果真有能确定的牛人 就内存拷贝最简单),自己进行字符串解析,不会有多少行代码
cobra_chen 2012-03-31
  • 打赏
  • 举报
回复
1.内存拷贝,楼上已经说了。
2.
Rec rec = *(Rec *)ss;
理论上应该可以。
-
但是不建议用强制转换。
楼主这个是32位的环境,如果是16位或64位的环境,很明显数据就错乱了。
大量的代码,这样的错误不好找。
sxsong_04 2012-03-31
  • 打赏
  • 举报
回复
[Quote=引用 3 楼 的回复:]
不能直接memcpy
需要针对每个结构域 从串 char*ss = "0001aaaaabbbbbbcccccccmssssssssss00030006";
中解析出来。
[/Quote]
他的字符串估计是从Rec类型转的 应该不是手写吧
gameslq 2012-03-31
  • 打赏
  • 举报
回复
不能直接memcpy
需要针对每个结构域 从串 char*ss = "0001aaaaabbbbbbcccccccmssssssssss00030006";
中解析出来。
sxsong_04 2012-03-31
  • 打赏
  • 举报
回复
int size = strlen(ss);
Rec * p = new Rec;
memcpy(p, ss, size);
hen_hao_ji 2012-03-31
  • 打赏
  • 举报
回复
内存拷贝 memcpy();

65,184

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧