51单片机实现串口通信

文章介绍了串行接口的基本概念,包括串口的定义、特点以及常见的RS-232、RS-422和RS-485标准。讨论了UART异步串行通信的工作方式和电平标准,如TTL和RS232电平。此外,还详细说明了单片机通过串口通信配置波特率的过程,并展示了如何通过串口控制LED灯的代码示例,包括使用中断提高响应速度的方法。
摘要由CSDN通过智能技术生成

串口基本认知

串行接口简称串口,也称串行通信接口或串行通讯接口(通常指COM接口),是采用串行通信方式的扩展接口。串行接口(Serial Interface)是指数据一位一位地顺序传送。其特点是通信线路简单,只要一对传输线就可以实现双向通信(可以直接利用电话线作为传输线),从而大大降低了成本,特别适用于远距离通信,但传送速度较慢

特点:
是设备间接线通信的一种方式
数据一位一位地顺序传送
双向通信,全双工
传送速度相对较慢

RS-232
也称标准串口,最常用的一种[串行通讯接口,比如我们的电脑主机的9针串口 ,最高速率为20kb/s
RS-232是为点对点(即只用一对收、发设备)通讯而设计的,其传送距离最大为约15米。所以RS-232适
合本地设备之间的通信

RS-422
由于接收器采用高输入阻抗和发送驱动器比RS232更强的驱动能力,故允许在相同传输线上连接多个接
收节点,最多可接10个节点。即一个主设备(Master),其余为从设备(Slave),从设备之间不能通
信,所以RS-422支持点对多的双向通信。
RS-422的最大传输距离为1219米,最大传输速率为10Mb/s。平衡双绞线的长度与传输速率成反比
RS-485
是从RS-422基础上发展而来的,无论四线还是二线连接方式总线上可多接到32个设备。

关于串口的电平
经常听说的UART
异步串行是指UART(Universal Asynchronous Receiver/Transmitter),通用异步接收/发送。
UART包含TTL电平的串口和RS232电平的串口
RS232电平
逻辑1为-3~-15V的电压, 逻辑0为3~15V的电压
笔记本通过RS232电平和单片机通信

TTL电平
TTL是Transistor-Transistor Logic,即晶体管-晶体管逻辑的简称,它是计算机处理器控制的设备
内部各部分之间通信的标准技术。TTL电平信号应用广泛,是因为其数据表示采用二进制规定,
+5V等价于逻辑”1”,0V等价于逻辑”0”。
数字电路中,由TTL电子元器件组成电路的电平是个电压范围,规定:
输出高电平>=2.4V,输出低电平<=0.4V;
输入高电平>=2.0V,输入低电平<=0.8V
笔记本电脑通过TTL电平与单片机通信
TX发送线(端口)3.1
RX接收线 (端口)3.0

串口接线方式
RXD:数据输入引脚,数据接受;
TXD:数据发送引脚,数据发送;

接线图:
在这里插入图片描述

串口编程要素:

输入/输出数据缓冲器都叫做SBUF, 都用99H地址码,但是是两个独立的8位寄存器
代码体现为: 想要接收数据 char data = SBUF 想要发送数据 SBUF = data

在这里插入图片描述

回忆UART是异步串行接口,通信双方使用时钟不同,因为双方硬件配置不同,但是需要约定通信速度,叫做波特率
对于电脑来说,别人做好了软件,鼠标点点点就能配置好,而苦逼单片机的波特率配置需要我们写代码配置什么,我们代码也要配置对应参数

关于如何配置波特率和串口通信,自己可以根据单片机开发手册进行:

利用串口通信发送字符实现对LED的控制:代码如下:

#include “reg52.h”
#include “intrins.h”

sfr AUXR = 0x8E;
sbit D5 = P3^7;

void UartInit(void) //9600bps@11.0592MHz
{
AUXR = 0x01;//降低电磁干扰
SCON = 0x50; //配置串口工作方式1,REN使能接收
TMOD &= 0x0F;
TMOD |= 0x20;//定时器1工作方式位8位自动重装

TH1 = 0xFD;
TL1 = 0xFD;//9600波特率的初值
TR1 = 1;//启动定时器1

}

void Delay1000ms() //@11.0592MHz
{
unsigned char i, j, k;

_nop_();
i = 8;
j = 1;
k = 243;
do
{
	do
	{
		while (--k);
	} while (--j);
} while (--i);

}

void sendByte(char data_msg)
{
SBUF = data_msg;
while(!TI);//串口发送控制位
TI = 0;
}

void sendString(char* str)
{
while( *str != ‘\0’){
sendByte(*str);
str++;
}
}

void main()
{

char *p = "xfj handsome!\r\n";
char cmd;
D5 = 1;
//配置C51串口的通信方式
UartInit();

while(1){
	
	Delay1000ms();
	//往发送缓冲区写入数据,就完成数据的发送
	sendString(p);
	//怎么知道收到数据,查询RI的值,如果RI是1(收到数据后由硬件置1)
	if(RI == 1){//串口接收控制位
		RI = 0;
		cmd = SBUF;//将指令写入单片机
		if(cmd == 'o'){
			D5 = 0;//点亮D5
		}
		if(cmd == 'c'){
			D5 = 1;//熄灭D5
		}
		
	}
}

}

结果图如下:
没发送字符控制开灯时:
在这里插入图片描述
在这里插入图片描述

用串口发送开灯时:
在这里插入图片描述
在这里插入图片描述

但发现点灯和关灯的反应不是那么灵敏,所以采取串口中断的方式提高它的响应速度,代码如下:

#include “reg52.h”
#include “intrins.h”

sfr AUXR = 0x8E;
sbit D5 = P3^7;

char cmd;

void UartInit(void) //9600bps@11.0592MHz
{
AUXR = 0x01;//降低电磁干扰
SCON = 0x50; //配置串口工作方式1,REN使能接收
TMOD &= 0x0F;
TMOD |= 0x20;//定时器1工作方式位8位自动重装

TH1 = 0xFD;
TL1 = 0xFD;//9600波特率的初值
TR1 = 1;//启动定时器1

EA = 1;//开启中断总开关
ES = 1;//开启串口中断开关

}

void Delay1000ms() //@11.0592MHz
{
unsigned char i, j, k;

_nop_();
i = 8;
j = 1;
k = 243;
do
{
	do
	{
		while (--k);
	} while (--j);
} while (--i);

}

void sendByte(char data_msg)
{
SBUF = data_msg;
while(!TI);//串口发送控制位
TI = 0;
}

void sendString(char* str)
{
while( *str != ‘\0’){
sendByte(*str);
str++;
}
}

void main()
{

char *p = "xfj handsome!\r\n";

D5 = 1;
//配置C51串口的通信方式
UartInit();

while(1){
	
	Delay1000ms();
	//往发送缓冲区写入数据,就完成数据的发送
	sendString(p);
	
}

}

void UART_Handler() interrupt 4
{
//怎么知道收到数据,查询RI的值,如果RI是1(收到数据后由硬件置1)
if(RI){//中断处理函数中对于接收中断的响应,当RI等于1时才执行下面的代码
RI = 0;//复位接收中断标志
cmd = SBUF;//将指令写入单片机
if(cmd == ‘o’){
D5 = 0;//点亮D5
}
if(cmd == ‘c’){
D5 = 1;//熄灭D5
}

}
if(TI); //这里先把发送中断写出来,因为没有用上,先不做处理

}

效果和上面是一样的,就不发图了

串口发送字符串控制LED灯,代码如下:

#include “reg52.h”
#include “intrins.h”
#include “String.h”

#define SIZE 12
sfr AUXR = 0x8E;
sbit D5 = P3^7;

char cmd[SIZE];

void UartInit(void) //9600bps@11.0592MHz
{
AUXR = 0x01;//降低电磁干扰
SCON = 0x50; //配置串口工作方式1,REN使能接收
TMOD &= 0x0F;
TMOD |= 0x20;//定时器1工作方式位8位自动重装

TH1 = 0xFD;
TL1 = 0xFD;//9600波特率的初值
TR1 = 1;//启动定时器1

EA = 1;//开启中断总开关
ES = 1;//开启串口中断开关

}

void Delay1000ms() //@11.0592MHz
{
unsigned char i, j, k;

_nop_();
i = 8;
j = 1;
k = 243;
do
{
	do
	{
		while (--k);
	} while (--j);
} while (--i);

}

void sendByte(char data_msg)
{
SBUF = data_msg;
while(!TI);//串口发送控制位
TI = 0;
}

void sendString(char* str)
{
while( *str != ‘\0’){
sendByte(*str);
str++;
}
}

void main()
{

char *p = "xfj handsome!\r\n";

D5 = 1;
//配置C51串口的通信方式
UartInit();

while(1){
	
	Delay1000ms();
	//往发送缓冲区写入数据,就完成数据的发送
	sendString(p);
	
}

}

void UART_Handler() interrupt 4
{
static int i = 0;//静态变量,只初始化一次

if(RI){//中断处理函数中对于接收中断的响应,当RI等于1时才执行下面的代码
	RI = 0;//复位接收中断标志
	
	cmd[i] = SBUF;//将指令写入单片机
	i++;
	if(i == SIZE){
		i=0;
	}
	if(strstr(cmd,"en")){
		D5 = 0;//点亮D5
		i=0;
		memset(cmd,'\0',SIZE);//清空数组,方便接收下一条指令
	}
	if(strstr(cmd,"se")){
		D5 = 1;//熄灭D5
		i=0;
		memset(cmd,'\0',SIZE);
	}
		
}
if(TI); 

}

效果的图如下:
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值