摘自:https://www.stmcu.org.cn/module/forum/thread-615994-1-17.html
} RecData; RecData userData; 结构体已经定义好了,接下来我们就将串口接收的数据复制到这个结构体 userData里面. 定义串口接收的缓冲数组为 U8 serRec[len]; U8 len=100; //按照实际接收范围定义
我这里使用的是stm32单片机,串口具有空闲中断功能,当检测到串口收到一帧数据后,会进入中断置位,此时我们将进行数据的复制,如下: memcpy(&userData,&serRec,len ); //内存复制 这就将串口数据完整的复制到userData中了, 而且是对号入座的, 错,stm32单片机是32位单片机,结构体中的变量存储时不是按照一字节对齐存储的,即默认每个存储单元分配是4个字节,像u8类型只占一字节这种默认也占了四个字节,显然空了3个字节,那么整个结构体的容量就不是结构体内所有变量的实际类型所占的大小之和. 我们要做的就是让结构体内的变量可以以一字节为最小单位对齐,即各个变量紧密的连在一起在内存中,这就需要字节对齐的预处理指令:
另外需要注意的是STM32的数据存储方式是小端模式,即数据的高字节存储在内存地址的高字节,低字节存储在内存地址的低字节,如果串口接收到的数据是低字节在前,则拷贝来的数据就不用进行处理,如果接收的数据是高字节在前则需要对拷贝来的数据进行高低字节反转.什么是大端和小端 Big-Endian和Little-Endian的定义如下: 1) Little-Endian就是低位字节排放在内存的低地址端,高位字节排放在内存的高地址端。 2) Big-Endian就是高位字节排放在内存的低地址端,低位字节排放在内存的高地址端。 举一个例子,比如数字0x12 34 56 78在内存中的表示形式为:
1)大端模式:
低地址 -----------------> 高地址 0x12 | 0x34 | 0x56 | 0x78
2)小端模式:
低地址 ------------------> 高地址 0x78 | 0x56 | 0x34 | 0x12
可见,大端模式和字符串的存储模式类似。
//keil 下1字节对齐 #pragma pack(push,1) typedef struct { u8 head; //数据头 u16 dataLen; //数据长度值 u8 height; //身高 u8 weight; //体重 u8 sex; //性别 u8 age; //年龄 u8 education; //学历 u8 checksum; //校验和 u8 endmark; //结束字节
} RecData;
#pragma pack(pop)//恢复keil原来的数据对齐方式
用这条预处理指令将结构体声明在其中即可,代表结构体中变量会在内存中按照一字节对齐的方式存储. 至此, userData.head userData.dataLen userData.height ......... 这些都已对号入座的数据就可以直接使用了.