串口接收不定长数据的几种方式

在阅读本文前,你需要先做到串口成功接收一个数据(相信这一点是很简单的) 
这几天简单总结了一下用串口怎么接收一帧数据的办法,个人使用的有三种,下面逐一介绍:

第一种:使用中断的方式;

这种在数据接收不频繁状态下使用。简单易实现。既然是使用中断,所以自然需要硬件的支持,比如stm32就可以这么做,具体操作见我之前的博客stm32串口中断接收一帧数据

8位单片机估计是没有的,嵌入式32位的处理器应该会有,可以依据自己使用的芯片查一下对应的寄存器手册,重点关注一下中断寄存器和uart寄存器的介绍部分。

第二种:使用自定义结束符;

这种也是很高效的,具体的做法是,在一帧数据尾部添加一个小尾巴 ~ ~ 
拿我的做法举例,因为我传输的内容里面不会出现特殊字符,所以我选择特殊字符 ‘#’标识一帧数据接收完成,然后立一个flag,交由通讯处理函数去处理。 
这里结束符的选取,不可以是通讯中可能传输的字符,尽量使用一个字符,简化逻辑判断。 
这种方式缺点也是很明显的,即只适用于自定义的通讯格式,或者尾部结束符唯一确定的情况。代码如下:

#define MAX_LEN 10  //单个数据包的最大长度
unsigned char rxDataBuff[MAX_LEN]; //串口数据接收区
unsigned char rxLen = 0; //串口接收数据计数
unsigned char rxFlag = 0;                     
//*************************************************
// 串口中断服务程序
// 依据你具体使用的mcu对应修改,这里只给出一个逻辑
// params:none
// return: none
// note:修改了全局变量 rxFlag 、 rxDataBuff[MAX_LEN] 、reLen
//*************************************************
void serialHandle()
{
    unsigned char dataTmp;
    if(RI == INT->status)//判断是否为接收中断
    {
        dataTmp = INT->data; //将串口buff的数据读出
        if('#' != dataTmp)
        {
            //
            //非结束符,把数据放到接收区
            //
            rxDataBuff[reLen++] = dataTmp;
        }
        else
        {
            //
            //结束符
            //
            rxDataBuff[reLen] = '\0';
            rxFlag = 1;
        }

    }
}
//*************************************************
// 循环内处理函数
// 此函数应该放到你的主循环里面
// 
// note:检测到flag首先清除标志位和接收计数
//*************************************************
void serialLoop()
{
    if(rxFlag == 1)
    {
        rxFlag = 0; //清除标志位
        rxLen = 0; //清除接收计数
        //
        //通讯处理
        //
    }
}

第三种:外加一个定时器

这种方式相对来说延时自己计算,具体做法是在串口接收数据时启动定时器,每接收一帧数据要复位定时值以保证定时器不会溢出。亲测在高频率接收不会乱码等。定时器的超时值设置为1.5倍接收一个数据的时间,这样一来,当定时器超时时候,就意味着串口接收数据停止了。此时同样立一个flag,然后交由通讯处理函数处理。 
这里的定时器大可不必单独再用一个定时器,可以使用系统滴答定时器,让滴答一次计数加一。超时的值设置要根据波特率调整,例如9600的波特率下,接收数据速率为9600bit/s = 1200B/s 也就是 0.83ms/B 那么我们的超时值可以设置为0.84*1.5 = 1.26ms 取个整1.3ms。其他波特率以此类推…… 
如果数据过去频繁,这种方式可能会产生粘包的情况(没测,如果有这方面使用需求一定要自己测试一下)。关于粘包处理,如果每个包数据长度基本相等的话,可以简单做一个根据包长度判断是否发生粘包。

#define MAX_LEN 10  //单个数据包的最大长度
unsigned char rxDataBuff[MAX_LEN]; //串口数据接收区
unsigned char rxLen = 0; //串口接收数据计数
unsigned char rxFlag = 0;
volatile unsigned char usCnt = 0; // 滴答计数变量,必须用volatile修饰
volatile unsigned char rxStart = 0; //开始接收标识
//*************************************************
// 串口中断服务程序
// 依据你具体使用的mcu对应修改,这里只给出一个逻辑
// params:none
// return: none
// note:修改了全局变量 rxFlag 、 rxDataBuff[MAX_LEN] 、reLen
//*************************************************
void serialHandle()
{
    unsigned char dataTmp;
    if(RI == INT->status)//判断是否为接收中断
    {
        rxStart = 1;
        rxDataBuff[rxLen++] = INT->data;
        usCnt = 0;
    }
}
//*************************************************
// 循环内处理函数
// 此函数应该放到你的主循环里面
// 
// note:检测到flag首先清除标志位和接收计数
//*************************************************
void serialLoop()
{
    if(rxFlag == 1)
    {
        rxFlag = 0; //清除标志位
        rxLen = 0; //清除接收计数
        //
        //通讯处理…………
        //
    }
}
//*************************************************
// 串口超时检测函数
// 此函数应该放到你系统滴答定时器处理函数里面
// 我的滴答设置是500us中断一次
// note:
//*************************************************
void serialTimeOut()
{
    if(rxStart == 1)
    {
        if(++usCnt > 3)
        {
            rxStart = 0;
            rxFlag = 1;
        }
    }
}

这就是我目前所能想到和已经实践过的三种串口接收一帧数据的方式了,如果觉得有用就顶一下吧

       ,___          .-;'
        "-.`\_...._/.`
    ,      \        /
 .-' ',    / ()   ()\
`'._   \  /()    .  (|
    > .' ;,     -'-  /
   / <   |;,     __.;
   '-.'-.|  , \    , \
      `>.|;, \_)    \_)
       `-;     ,    /
          \    /   <
           '. <`'-,_)
            '._)
stm32cubemx是一款适用于STM32微控制器的图形化配置工具,可以帮助开发人员快速配置各种外设及功能,并生成相应的初始化代码。如果需要在STM32微控制器上实现串口接收不定数据,可以通过使用stm32cubemx来配置串口外设,并结合中断来实现。 首先,在stm32cubemx中选择需要的串口外设,并配置相应的参数,如波特率、数据位数、停止位数等。接着,在中断配置中使能串口接收中断并编写中断服务函数,当串口接收数据时,中断服务函数将被触发。 在中断服务函数中,可以通过轮询的方式或者使用缓冲区来接收不定数据。一种常见的做法是使用循环队列或者循环指针来缓存接收到的数据,当接收到新数据时,将数据写入缓冲区,并判断是否达到预定的数据度。当达到预定的数据度时,可以进行数据处理或者触发相应的事件。 另外,也可以结合DMA(直接内存存取)来实现串口接收不定数据,通过配置DMA通道来自动接收数据并存储到指定的缓冲区中,当接收到预定度的数据时,DMA将触发相应的中断或者事件,通知程序进行后续的处理。 综上所述,通过stm32cubemx的串口配置和中断/ DMA的使用,可以在STM32微控制器上实现串口接收不定数据的功能。开发人员可以根据具体需求和项目要求选择合适的方案来实现串口接收不定数据的功能。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值