51单片机串口接收处理

目的: 

通过51单片机串口外设功能,来对一组数据进行处理,新手刚刚入门可能会将串口接收到的一个数据直接赋值给一个变量,然后用这个变量做相应功能,可是这样写的不稳定性太高了,有可能会接收不到你要的那个数据,其次就是接受到错误的数据

功能:

(判断头,取接收长度,接收有效数据,接收校验位和计算校验)


步骤:

假设串口接收的这样一组数据


假设串口接收到这样一组数据,这个是规格书,实际串口接收到的数据和这个也是一样

 代码实现处理

void UART_ISR(void) interrupt 4
{

    UINT8 temp = 0;
    if (RI == 1)
    {
		RI = 0;                         // Clear reception flag for next reception
		temp = SBUF;
		switch(strUart0.status){
//-------------------------判断头字节是否正确----------------------------//
			case 0:
				if(temp == 0x42){
					strUart0.status = 1;
					strUart0.RxdBuf[strUart0.recIndex] = temp;
					strUart0.recIndex ++;					
					strUart0.ChkSum += temp;					
				}
				else{
					resetUart0();
				}
			break;
			case 1:
				if(temp == 0x4D){                                            
				strUart0.status = 2;
				strUart0.charRxCountLen = 0;
				strUart0.RxdBuf[strUart0.recIndex] = temp;
				strUart0.recIndex ++;
				strUart0.ChkSum += temp;
					
			  }
				else{
					resetUart0();
				}
			break;
//----------------------------------------------------------------------------

//-----------------------------------取接收长度-----------------------------				
			case 2:	
				strUart0.status = 3;
				strUart0.RxdBuf[strUart0.recIndex] = temp;
				strUart0.recIndex++;		
				strUart0.charRxCountLen = temp;
				strUart0.ChkSum += temp;				
			break;
				
			case 3:	
				strUart0.status = 4;
				strUart0.RxdBuf[strUart0.recIndex] = temp;
				strUart0.recIndex++;		
				strUart0.charRxCountLen = strUart0.charRxCountLen * 256 + temp;
				strUart0.ChkSum += temp;			
			break;
//-------------------------------------------------------------------------------

//-------------------------------------接收有效数据--------------------------------			
			case 4:
				strUart0.RxdBuf[strUart0.recIndex] = temp;
				strUart0.recIndex++;
				strUart0.ChkSum += temp;
				if(--strUart0.charRxCountLen == 2){
					strUart0.status = 5;
				}			
			break;
//---------------------------------------------------------------------------------		

//--------------------------------------接收校验位和计算校验-------------------------				
			case 5:
				strUart0.RxdBuf[strUart0.recIndex] = temp;
				strUart0.recIndex++;
				if(strUart0.recIndex > UART1_RXDBUF_MAX){
					strUart0.recIndex = 0;
				}
				if(--strUart0.charRxCountLen ==0){
					if(strUart0.ChkSum == (unsigned short)strUart0.RxdBuf[strUart0.recIndex-2] * 256 + strUart0.RxdBuf[strUart0.recIndex-1]){
						strUart0.recFlagOK = 1;							
						strUart0.status = 0;
						strUart0.recIndex = 0;
					}
					else{
						resetUart0();
					}
				}				
			break;
//-------------------------------------------------------------------------------------
				
			default:
				resetUart0();
				break;
		}
    }
}
  while (1)
    {
					
//		Timer0_Delay1ms(10);

		if(strUart0.recFlagOK == 1){
			LD15_Dust_Value = (unsigned short)strUart0.RxdBuf[12]*256 + strUart0.RxdBuf[13];
			resetUart0();
		}
		
    }

strUart0.RxdBuf[strUart0.recIndex-2] * 256 + strUart0.RxdBuf 


注意:

这里 *  256 相当于 <<8   (左移一位是2^n次方,一共左移8 也就是2的8次方)

这里找到一个很好理解的方法

0x1是16进制整数,等于10进制的1。

0x1<<2表示将1左移两位后的结果。

1左移两位后的2进制表示是100,等于10进制的4。

因为一个整数左移n位的结果就相当于乘以2^n,所以,0x1<<2的结果也可以这样计算得出:1*2^2等于4。

代码及运行结果如下图:

 

  • 6
    点赞
  • 33
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
51单片机串口接收数据程序需要进行以下步骤: 1. 初始化串口,包括设置波特率、数据位、停止位等参数。 2. 开启串口接收中断,以便在接收到数据时能够及时处理。 3. 在中断服务函数中,读取接收缓冲区中的数据,并进行相应的处理。 4. 在主函数中,可以通过查询接收缓冲区是否有数据来判断是否接收到了数据。 下面是一个简单的51单片机串口接收数据程序的示例: ``` #include <reg52.h> // 定义串口参数 #define BAUDRATE 9600 #define DATA_BITS 8 #define STOP_BITS 1 // 定义接收缓冲区 #define BUFFER_SIZE 32 unsigned char buffer[BUFFER_SIZE]; unsigned char buffer_head = 0; unsigned char buffer_tail = 0; // 串口中断服务函数 void serial_isr() interrupt 4 { if (RI) { // 接收中断 RI = 0; buffer[buffer_tail] = SBUF; buffer_tail = (buffer_tail + 1) % BUFFER_SIZE; } } // 初始化串口 void init_serial() { TMOD |= 0x20; // 设置定时器1为模式2 TH1 = 256 - FOSC / 12 / BAUDRATE; // 计算波特率 TL1 = TH1; PCON &= 0x7F; // 波特率不加倍 SCON = 0x50; // 设置串口为模式1 ES = 1; // 开启串口中断 EA = 1; // 开启总中断 } // 查询接收缓冲区是否有数据 unsigned char serial_available() { return buffer_head != buffer_tail; } // 读取接收缓冲区中的数据 unsigned char serial_read() { unsigned char data = buffer[buffer_head]; buffer_head = (buffer_head + 1) % BUFFER_SIZE; return data; } void main() { init_serial(); while (1) { if (serial_available()) { unsigned char data = serial_read(); // 处理接收到的数据 } } } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

长生君

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值