51单片机-串口

1、串口简介

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

——是设备间接线通信的一种方式

——数据一位一位地顺序传送

——双向通信,全双工

——传送速度相对较慢

补充(全双工与半双工的区别):

全双工(Full Duplex)通信允许数据在两个方向上同时传输,它在能力上相当于两个单工通信方式的结合。全双工指可以同时(瞬时)进行信号的双向传输(A→B且B→A)。指A→B的同时B→A,是瞬时同步的。

半双工(Half Duplex)数据传输指数据可以在一个信号载体的两个方向上传输,但是不能同时传输。

2、串口协议

串行接口按电气标准及协议来分包括RS-232,RS-422,RS485等。以上只对接口的电气特性做出规定,不涉及接插件、电缆或协议。

①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个设备。

3、串口电平

        经常听说的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,输出低电平=2.0V,输入低电平<=0.8V

笔记本电脑通过TTL电平与单片机通信                                                                                           TX发送线(端口)————P3.1                                                                                                       RX接收线  (端口)————P3.0 

4、串口通信

4.1串口接线

RXD:数据输入引脚,数据接收;STC89系列对应P3.0口

TXD:数据发送引脚,数据发送;STC89系列对应P3.1口

4.2串口编程

数据传输速度:波特率       校验位         停止位

4.3寄存器

1、SBUF

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

2、PCON

SMOD:波特率选择位, SMOD=1,串行通信方式1、2、3的波特率加倍;SMOD =0,则不加倍

3、SCON

4、实验一:发送字母给PC

#include "reg52.h"

sfr AUXR = 0x8e;

//void UartInit(void)		//9600bps@11.0592MHz
//{
//	PCON &= 0x7F;		//波特率不倍速
//	SCON = 0x50;		//8位数据,可变波特率
//	AUXR &= 0xBF;		//定时器1时钟为Fosc/12,即12T
//	AUXR &= 0xFE;		//串口1选择定时器1为波特率发生器
//	TMOD &= 0x0F;		//清除定时器1模式位
//	TMOD |= 0x20;		//设定定时器1为8位自动重装方式
//	TL1 = 0xFD;		//设定定时初值
//	TH1 = 0xFD;		//设定定时器重装值
//	ET1 = 0;		//禁止定时器1中断
//	TR1 = 1;		//启动定时器1
//}
void UartInit(void)
{
	SCON = 0x40;//配置串口工作方式1,REN不使能接收
	TMOD &= 0x0F;//高四位清零
	TMOD |= 0x20;//定时器1工作方式8位自动重装
	AUXR = 0x01;
	
	TL1 = 0xFD;		//设定定时初值
	TH1 = 0xFD;	//9000波特率的初值
	TR1 = 1;		//启动定时器1
}

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

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

//发送1个字符
void sendByte(char data1)
{
	SBUF = data1;
}

void main()
{
	char data1 = 'w';
	//配置C51串口的通信方式
	UartInit();
	
	while(1){
		Delay1000ms();
		//往发送缓冲区写入数据来完成数据的发送
		sendByte(data1);
		
	}
}

实验二:发送字符串给PC

#include "reg52.h"

sfr AUXR = 0x8e;

//void UartInit(void)		//9600bps@11.0592MHz
//{
//	PCON &= 0x7F;		//波特率不倍速
//	SCON = 0x50;		//8位数据,可变波特率
//	AUXR &= 0xBF;		//定时器1时钟为Fosc/12,即12T
//	AUXR &= 0xFE;		//串口1选择定时器1为波特率发生器
//	TMOD &= 0x0F;		//清除定时器1模式位
//	TMOD |= 0x20;		//设定定时器1为8位自动重装方式
//	TL1 = 0xFD;		//设定定时初值
//	TH1 = 0xFD;		//设定定时器重装值
//	ET1 = 0;		//禁止定时器1中断
//	TR1 = 1;		//启动定时器1
//}
void UartInit(void)
{
	SCON = 0x40;//配置串口工作方式1,REN不使能接收
	TMOD &= 0x0F;//高四位清零
	TMOD |= 0x20;//定时器1工作方式8位自动重装
	AUXR = 0x01;
	
	TL1 = 0xFD;		//设定定时初值
	TH1 = 0xFD;	//9000波特率的初值
	TR1 = 1;		//启动定时器1
}

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

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

//发送1个字符
void sendByte(char data1)
{
	SBUF = data1;
	while(!TI);//发送中断请求标志位,发送完成变为1,取反为0不成立则继续执行下行代码
	TI = 0;
}
//发送字符串
void sendString(char* str)
{
	while(*str != '\0'){				//字符串末尾为\0
		sendByte(*str);
		str++;
	}
}
void main()
{
	//配置C51串口的通信方式
	UartInit();
	
	while(1){
		Delay1000ms();
		//往发送缓冲区写入数据来完成数据的发送
		sendString("Hello World\r\n");
	}
}

实验三:PC端发送指令控制LED

REN=1:容许接收数据

RI:接收中断标志位,收到数据时,RI=1(硬件置1),必须软件置位0

#include "reg52.h"

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

//void UartInit(void)		//9600bps@11.0592MHz
//{
//	PCON &= 0x7F;		//波特率不倍速
//	SCON = 0x50;		//8位数据,可变波特率
//	AUXR &= 0xBF;		//定时器1时钟为Fosc/12,即12T
//	AUXR &= 0xFE;		//串口1选择定时器1为波特率发生器
//	TMOD &= 0x0F;		//清除定时器1模式位
//	TMOD |= 0x20;		//设定定时器1为8位自动重装方式
//	TL1 = 0xFD;		//设定定时初值
//	TH1 = 0xFD;		//设定定时器重装值
//	ET1 = 0;		//禁止定时器1中断
//	TR1 = 1;		//启动定时器1
//}
void UartInit(void)
{
	SCON = 0x50;//配置串口工作方式1,REN使能接收
	TMOD &= 0x0F;//高四位清零
	TMOD |= 0x20;//定时器1工作方式8位自动重装
	AUXR = 0x01;
	
	TL1 = 0xFD;		//设定定时初值
	TH1 = 0xFD;	//9000波特率的初值
	TR1 = 1;		//启动定时器1
}

void main()
{
	char cmd;
	D5 = 1;
	//配置C51串口的通信方式
	UartInit();
	
	while(1){
		if(RI == 1){
			RI = 0;//软件置位
			cmd = SBUF;
			if(cmd == 'o'){
				D5 = 0;
			}
			if(cmd == 'c'){
				D5 = 1;
			}
		}
	}
}

实验四:PC通过串口中断控制LED

中断函数开灯,开启允许串口中断和总中断

#include "reg52.h"

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

char cmd;

void UartInit(void)
{
	SCON = 0x50;//配置串口工作方式1,REN使能接收
	TMOD &= 0x0F;//高四位清零
	TMOD |= 0x20;//定时器1工作方式8位自动重装
	AUXR = 0x01;
	
	TL1 = 0xFD;		//设定定时初值
	TH1 = 0xFD;	//9000波特率的初值
	TR1 = 1;		//启动定时器1
	
	ES = 1;//允许串口中断开启
	EA = 1;//总中断打开
}

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

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

//发送1个字符
void sendByte(char data1)
{
	SBUF = data1;
	while(!TI);//发送中断请求标志位,发送完成变为1,取反为0不成立则继续执行下行代码
	TI = 0;
}
//发送字符串
void sendString(char* str)
{
	while(*str != '\0'){				//字符串末尾为\0
		sendByte(*str);
		str++;
	}
}

void main()
{
	D5 = 1;
	//配置C51串口的通信方式
	UartInit();
	
	while(1){
		Delay1000ms();
		//往发送缓冲区写入数据来完成数据的发送
		sendString("Hello World\r\n");
	}
	
}

void Uart_Handler() interrupt 4
{
	if(RI == 1)               //中断处理函数中,对于接收中断的响应
	{
		RI = 0;
		cmd = SBUF;
		if(cmd == 'o'){
			D5 = 0;
		}
		if(cmd == 'c'){
			D5 = 1;
		}
	}
	if(TI ==1);               //中断处理函数中,对于发送中断的响应
}

实验五:串口通过单词型指令控制

单个字符不行,通过数组

#include "reg52.h"
#include <string.h>

#define SIZE 12        //宏定义

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

char cmd[SIZE];        //数组

void UartInit(void)
{
	SCON = 0x50;//配置串口工作方式1,REN使能接收
	TMOD &= 0x0F;//高四位清零
	TMOD |= 0x20;//定时器1工作方式8位自动重装
	AUXR = 0x01;
	
	TL1 = 0xFD;		//设定定时初值
	TH1 = 0xFD;	//9000波特率的初值
	TR1 = 1;		//启动定时器1
	
	ES = 1;//允许串口中断开启
	EA = 1;//总中断打开
}

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

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

//发送1个字符
void sendByte(char data1)
{
	SBUF = data1;
	while(!TI);//发送中断请求标志位,发送完成变为1,取反为0不成立则继续执行下行代码
	TI = 0;
}
//发送字符串
void sendString(char* str)
{
	while(*str != '\0'){				//字符串末尾为\0
		sendByte(*str);
		str++;
	}
}

void main()
{
	D5 = 1;
	//配置C51串口的通信方式
	UartInit();
	
	while(1){																	//心脏包,判断单片机是否正常工作
		Delay1000ms();
		//往发送缓冲区写入数据来完成数据的发送
		sendString("Hello World\r\n");
	}
	
}

void Uart_Handler() interrupt 4
{
	static int i = 0;      //静态变量,初始化一次
	if(RI == 1)               //中断处理函数中,对于接收中断的响应
	{
		RI = 0;
		cmd[i] = SBUF;
		i++;
		//如果到尾则重头来
		if(i == SIZE){
			i = 0;
		}
		//判断指令
		if(strstr(cmd,"en")){			//如果指令单词中包含en则执行
			D5 = 0;
			i = 0;
			memset(cmd,'\0',SIZE);	//清空数组
		}
		if(strstr(cmd,"se")){			//如果指令单词中包含se则执行
			D5 = 1;
			i = 0;
			memset(cmd,'\0',SIZE);
		}
	}
	if(TI ==1);
}

  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 学习51单片机和C语言编程,可以帮助我们更深入地理解嵌入式系统的原理和工作方式。对于初学者来说,掌握一份适合自己的学习资料非常重要。 要学习51单片机-C语言版,可以阅读《手把手教你学51单片机-C语言版pdf》这本电子书,这本书内容丰富,讲解详细,合实例编程,非常适合初学者自学。以下是学习本书的几个关键点: 第一,掌握基本的硬件知识,包括单片机的结构和特性,尤其是各种寄存器的作用和置方法。 第二,了解C语言编程基础,尤其是语法、数据类型、运算符、控制结构、函数等,这是编写单片机程序的基础。 第三,通过实例编程加强对知识的理解和运用能力。例如,可以尝试写一些简单的IO控制、定时器中断、串口通讯等程序。 第四,可以搭相应的开发板和开发环境进行实践学习。例如,可以使用STC89C51开发板和Keil或SDCC开发环境。 总之,《手把手教你学51单片机-C语言版pdf》这本电子书是一个不错的学习资料,但也需要具备一定的基础知识和耐心,可以结合其他资料和实践不断提高自己的能力。 ### 回答2: 学习51单片机-c语言版, 需要基础的C语言编程知识。在学习前,先要熟悉C语言的数据类型、循环、判断及函数等语法结构,并掌握C语言的编写方法。 在学习51单片机-c语言版之前,需要准备好学习环境,如下载并安装Keil软件, 安装并关联好相应的单片机模拟器。Keil软件中有类似于记事本的编辑窗口用来编写C语言代码, 以及编译,调试和下载程序到单片机等功能。 在学习时,可以选择一些简单的例程开始学习,逐步理解其代码逻辑,了解基本的寄存器操作和中断等知识。可以从LED灯等简单的实验开始,逐渐增加难度和功能的复杂度。 同时,可以参考一些权威的学习资料如《单片机原理与应用》、《51单片机学习与应用》等相关书籍,或结合网络资源进行学习。在学习过程中,需要勤加练习,多编写代码进行实践,同时多与他人交流学习体会和技术问题。通过坚持不断的学习和练习,便可以逐步掌握51单片机-c语言版编程技巧,提高自己的单片机应用开发能力。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值