我之前写过一篇文章关于STM32单片机接收不定长数据的方法,这篇文章可以在瑞生网(https://Mrs.Pub)搜索关键字“不定长”看到,不过...
这个只是针对某些STM32单片机的方法,比如F1系列的就可以使用,然而...
我在后续使用其它STM32系列的单片机时,发现这个方法不能用,比如L0系列的,我用IDLE判断,总是还没有接收完一帧数据,就会发生IDLE中断,于是...
我只能用下面的通用方法来解决了,这个通用的方法,其实原理和使用IDLE的原理一样:
接收完一个字节以后,如果超过了一定的时间,就认为是接收完一帧数据了。首先...
我们要知道,串口是接收一个字节,就会发生一次中断,如果一帧数据包含10个字节,就会发生10次中断。在接收一个字节以后,会紧跟着接收下一个字节,如果时间超了一定值,就代表一帧数据已经发完了。
下面,我分别用STM32和51单片机的代码来演示一下这个通用代码的实现。
void LPUART1_IRQHandler(void)
{
if(__HAL_UART_GET_FLAG(&hlpuart1, UART_FLAG_RXNE))
{
Res_Buf[Res_Count++]=hlpuart1.Instance->RDR;
Res_Sign=1;
Res_Times=0;
}
}
void UART1_Isr() interrupt 4
{
if(RI)
{
RI = 0;
Res_Buf[Res_Count++]=SBUF;
Res_Sign=1;
Res_Times=0;
}
}
if(Res_Sign==1)
{
do{
Res_Times++;
HAL_Delay(1);
}while(Res_Times<5);
Res_Sign=0;
Res_Count=0;
}
unsigned char Res_Buf[256];
unsigned char Res_Count=0;
unsigned char Res_Sign=0;
unsigned char Res_Times=0;
在串口中断函数里面,每接收到一个字节,就会把接收到的字节保存到Res_Buf数组中,同时,字节计数器+1,然后把Res_Sign置1,表示已经接收到串口数据,但是,有没有接收完,是不一定的。
在主函数当中,发现这个变量等于1了,就开始启动延时计数Res_Times,让这个变量++,只要延时到了5ms,就表示接收完一帧数据,退出do while后就可以开始处理数据了,但是...
当接收到第二个字节以后,会在中断函数里面把Res_Times清0,也就是说,主函数里面的Res_Times++以后,白加了,只要有数据还没有接收完,这个Res_Times就会一直清0。
如果串口一万年也接收不完一帧数据,那一万年,Res_Times也到不了5。只有当再也没有串口数据过来了,Res_Times才会加到5,然后退出do while,表示接收完一帧数据了,可以开始处理了。
上面的方法,适合所有的单片机。方法好不好,主要看自身产品的需求,适合就好,不适合就不好。方法有多种,根据你的需求调整到最好,就可以。