UART串口程序设计for蓝桥杯

九层妖塔 起于垒土

在这里插入图片描述



改编自国信长天蓝桥杯官方蓝皮书例程,按照自己的习惯进行了补充和修改


一、UART串口基本知识

 ●单片机和单片机直接,都是使用TTL电平可以直接通信
在这里插入图片描述
 ●单片机和PC直接,需要进行电平转换
在这里插入图片描述

 ●发送接收示意图:
在这里插入图片描述

二、需要记住的硬件寄存器

 ●STC15F2K60S2系列共2个采用UART工作方式的高速全双工异步串行通信接口。

 ●通用异步收发传输器(Universal Asynchronous Receiver/Transmitter),每个串行口由两个数据缓冲器、一个移位寄存器、一个串行控制寄存器、一个波特率发生器

 ●SCON (Serial Control)串行控制寄存器,地址0x98,可位寻址。
  ○ TI发送中断请求标志位
  ○ RI接收中断请求标志位
在这里插入图片描述

 ●SBUF(Serial Buffer),串行口数据缓冲寄存器。
  ○发送缓冲器和接收缓冲器共用一个地址。
  ○对SBUF寄存器进行写操作时,写的是只写寄存器–发送缓冲器,写操作完成待发送数据的加载。
  ○对SBUF寄存器进行读操作时,读的是只读寄存器–接收缓冲器,读操作可以获得已接收到的数据。

在这里插入图片描述

 ●ES串行口中断允许位
在这里插入图片描述

三、串口1工作模式1的工作过程及初始化配置

 ●受限于蓝桥杯开发板,常用串口和工作模式为串口1的工作模式1。

在这里插入图片描述
 ●工作模式1的发送过程:
在这里插入图片描述

 ●工作模式1的接收过程:

在这里插入图片描述

 ●用STC-ISP的波特率计算器生成初始化代码:
 使用定时器2作为波特率发生器,(定时器2不可位寻址,不方便作为定时器使用)
在这里插入图片描述

初始化代码:

void UartInit(void)		//4800bps@12.000MHz
{
	SCON = 0x50;		//8位数据,可变波特率
	AUXR |= 0x01;		//串口1选择定时器2为波特率发生器
	AUXR |= 0x04;		//定时器2时钟为Fosc,即1T
	T2L = 0x8F;		//设定定时初值
	T2H = 0xFD;		//设定定时初值
	AUXR |= 0x10;		//启动定时器2
}

四、代码

 ●发送采用轮询方式:单片机发送给PC机,是单片机控制将数据装入SBUF数据缓冲器,轮循到再发送,发送完这次的数据再将 下次的数据装入SBUF,不会造成数据覆盖。
 ●接收采用中断方式:单片机接收数据取决于上位机是否发送,上位机的发送具有随机性,单片机无法预知。因此,需要采用中断方法,即有新数据装入SBUF接收缓冲器时,单片机就立即读取SBUF中的数据,将其保存到变量中。避免单片机没有及时处理SBUF接收缓冲器中的数据,造成新数据将就数据覆盖。

 ●全局变量定义

//-------------------------------------------------UART
uchar puc_Uart_Send_String[x];  //串口发送缓冲数组
uchar uc_Uart_Rece_num;       //串口接收数据个数
uchar puc_Uart_Rece_String[x];  //串口接收缓冲数组

 ●串口中断服务程序

//UART 中断服务程序
void Uart_ISR(void) interrupt 4
{
  if(RI == 1) //接收完一个字节数据
	{
	  RI = 0;  //清除RI位
	  puc_Uart_Rece_String[uc_Uart_Rece_num++] = SBUF;  //保存串口数据
	}
}

 ●由于 发送完一个字节数据(RI被置1) 或者 接收完一个字节数据(TI被置1)都会触发串口中断,因此进入串口中断服务函数中,需要先判断是接收还是发送中断。并且需要手动软件清除标志位

 ●发送一个字节

void Uart_SendByte(unsigned char Byte)  //通过串口发送一个字节
{
	SBUF = Byte;     //将一个字节写入SBUF,开启发送
	while(TI == 0);  //等待发送完成
	TI = 0;
}

 ●发送一个字符串

void Uart_SendString(unsigned char *String)  //通过串口一个字符串
{
    while(*String != '\0')  //检测字符串结束标志
    {
	  Uart_SendByte(*String);//取 存储在 指针变量String上 的值给SBUF
      String++;
    }
}

 ●使用while(TI == 0);等待发送完成可以实现没有发送误差的连续发送。(STC-ISP上的例程貌似只能发送单个字符,发送字符串会出现误差)
 ●
在这里插入图片描述
 ●地址运算符 & : 后边跟一个变量名时,&给出该变量的地址。
 ●间接运算符 * :后边跟一个指针名或者地址时,*给出存储在 指针指向地址上的值。

 ●串口处理函数&串口接收&发送


void Uart_Proc(void)
{
	if(uc_Uart_Rece_num > 1)
	{
	  if( (puc_Uart_Rece_String[uc_Uart_Rece_num-1] == '\n') && (puc_Uart_Rece_String[uc_Uart_Rece_num-2] == '\r') )
		{
           if(uc_Uart_Rece_num == 4)
			{
				if( (puc_Uart_Rece_String[0] == 'S') && (puc_Uart_Rece_String[1] == 'T'))
				{
					uc_Uart_Rece_num = 0; //清零
				  Uart_Send_String("ST\r\n");
				}			  
			}
		    else if(uc_Uart_Rece_num == 6)
			{
				uc_Uart_Rece_num = 0; //清零
			    if( (puc_Uart_Rece_String[0] == 'P') && (puc_Uart_Rece_String[1] == 'A') && (puc_Uart_Rece_String[2] == 'R') && (puc_Uart_Rece_String[1] == 'A'))
			     {
			       Uart_Send_String("PARA\r\n");
			     }
			    else 
			    {
			      Uart_Send_String("ERROR_1\r\n");
			    }
		    }
			else 
			{
				uc_Uart_Rece_num = 0; //清零
				Uart_Send_String("ERROR_2\r\n");
			}				
		}
	
	}
  
}

Notes:

 ●当定义了一个大小为2的数组,使用时候索引为3,编译并不会报错,可实际运行结果是什么。待查证……

回车与换行
回车”(Carriage Return)和“换行”(Line Feed)这两个概念的来历和区别。
在Windows中:
‘\r’ 回车,回到当前行的行首,而不会换到下一行,如果接着输出的话,本行以前的内容会被逐一覆盖;
‘\n’ 换行,换到当前位置的下一行,而不会回到行首;

符号ASCII码意义
\n10换行 new line
\r13回车CR return

  
  
  
  
彩 蛋

  • 9
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

#法外狂徒张三

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

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

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

打赏作者

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

抵扣说明:

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

余额充值