博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
结构体在STM32串口接收中的妙用
阅读量:4082 次
发布时间:2019-05-25

本文共 1862 字,大约阅读时间需要 6 分钟。

 

摘自:

结构体在STM32串口接收中的妙用

结构体C程序中在STM32单片机串口接收的妙用之一
  结构体是一种数据的归类方式,相比数组或变量更具有整体全面性,例如一个数组只可以放一些按照元素顺序存放的单元变量,即tab
={x,x,x,x,x......},i
多大,数组内元素就有多少.那么我们这时候如果我们用这个数组来接收串口接收信息,信息的格式是: 数据头>数据长度>数据区>数据校验>数据尾.
假设数据区为 <身高-体重-性别-年龄-学历>
那么我们用数组接收时,提取数据时就需要计算出数据格式中每个单元所对应的位置,即数组中第i个元素对应的内容.这样显然是很麻烦的,效率很低.这就相当于先织了一个大网,捕捉到一网鱼,还得过下称,才能按照重量分类开来一样.
那么如果我们能提前根据接收的数据的格式来做一个容器,直接把接收的数据复制到这个容器内,岂不是省了好大劲,来,有个好东西,这就是结构体.
我们按照接收数据的格式顺序定义一个结构体如下:
typedef struct
{
         u8    head;            //数据头
        u16    dataLen;      //数据长度值
        u8    height;           //身高
         u8    weight;          //体重
         u8    sex;               //性别
         u8    age;               //年龄
         u8    education;     //学历
         u8    checksum;     //校验和
         u8    endmark;      //结束字节

 

} 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
.........
这些都已对号入座的数据就可以直接使用了.

转载地址:http://ralni.baihongyu.com/

你可能感兴趣的文章
network manager
查看>>
OS + Linux Disk disk lvm / disk partition / disk mount / disk io
查看>>
RedHat + OS CPU、MEM、DISK
查看>>
net TCP/IP / TIME_WAIT / tcpip / iperf / cain
查看>>
webServer kzserver/1.0.0
查看>>
OS + Unix IBM Aix basic / topas / nmon / filemon / vmstat / iostat / sysstat/sar
查看>>
my ReadMap subway / metro / map / ditie / gaotie / traffic / jiaotong
查看>>
OS + Linux DNS Server Bind
查看>>
linux下安装django
查看>>
Android 解决TextView设置文本和富文本SpannableString自动换行留空白问题
查看>>
Android开发中Button按钮绑定监听器的方式完全解析
查看>>
Android自定义View实现商品评价星星评分控件
查看>>
postgresql监控工具pgstatspack的安装及使用
查看>>
postgresql查看表的和索引的情况,判断是否膨胀
查看>>
postgresql中根据oid和filenode去找表的物理文件的位置
查看>>
postgresql减少wal日志生成量的方法
查看>>
swift中单例的创建及销毁
查看>>
获取App Store中App的ipa包
查看>>
iOS 关于pods-frameworks.sh:permission denied报错的解决
查看>>
设置tabbaritem的title的颜色及按钮图片
查看>>