基于STC12C5A60S2系列1T 8051单片机实现一主单片机发送数据给一从单片机接收并返回数据给主单片机的串口通信功能

基于STC12C5A60S2系列1T 8051单片机实现一主单片机发送数据给一从单片机接收并返回数据给主单片机的串口通信功能

STC12C5A60S2系列1T 8051单片机管脚图

在这里插入图片描述在这里插入图片描述# STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置在这里插入图片描述# STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍在这里插入图片描述在这里插入图片描述

STC12C5A60S2系列1T 8051单片机串口通信介绍

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

STC12C5A60S2系列1T 8051单片机串口通信的结构

基于STC12C5A60S2系列1T 8051单片机串口通信的特殊功能寄存器列表

在这里插入图片描述

基于STC12C5A60S2系列1T 8051单片机串口通信用到的特殊功能寄存器

STC12C5A60S2系列1T 8051单片机辅助寄存器AUXR

在这里插入图片描述
作用:用来设置STC12C5A60S2系列1T 8051单片机串口通信类型、波特率

STC12C5A60S2系列1T 8051单片机辅助寄存器AUXR1

在这里插入图片描述
作用:用来设置STC12C5A60S2系列1T 8051单片机串口通信引脚切换

串口1控制寄存器SCON

在这里插入图片描述
作用:用来设置STC12C5A60S2系列1T 8051单片机串口通信以下功能
(1)、设置串口通信工作模式
(2)、设置串口方式2或方式3多机通信
(3)、设置串口通信允许接收
(4)、设置串口通信在方式2或方式3下发送第9位数据
(5)、设置串口通信通信在方式2或方式3下接收第9位数据
(6)、设置串口通信发送中断请求
(7)、设置串口通信接收中断请求

串口电源控制寄存器PCON

在这里插入图片描述
作用:用来设置STC12C5A60S2系列1T 8051单片机串口通信以下功能
(1)、设置串口通信波特率选择
(2)、设置串口通信帧错误检测有效控制

串口数据缓冲寄存器SBUF

在这里插入图片描述在这里插入图片描述
作用:用来设置STC12C5A60S2系列1T 8051单片机串口通信发送或接收数据

串口辅助寄存器AUXR

在这里插入图片描述在这里插入图片描述
作用:用来设置STC12C5A60S2系列1T 8051单片机串口通信类型、波特率

串口中断寄存器

在这里插入图片描述
作用:用来设置STC12C5A60S2系列1T 8051单片机串口通信中断允许、中断优先级

基于STC12C5A60S2系列1T 8051单片机串口1通信工作模式

串口1通信模式0

在这里插入图片描述

串口1通信模式1

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

串口1通信模式2

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

串口1通信模式3

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

串口通信波特率设置

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

基于STC12C5A60S2系列1T 8051单片机实现一主单片机发送数据给一从单片机接收并返回数据给主单片机的串口通信功能介绍

串口通信简单介绍

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

基于STC12C5A60S2系列1T 8051单片机实现一主单片机发送数据给一从单片机接收并返回数据给主单片机的串口通信功能电路连接及工作原理

电路连接

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

工作原理

主单片机与从单片机成功通信后 主单片机先给从单片机发送数据 从单片机接收主单片机发送来的数据后 再把这个数据返回给主单片机 进行下一轮发送接收

基于STC12C5A60S2系列1T 8051单片机实现一主单片机发送数据给一从单片机接收并返回数据给主单片机的串口通信功能编程

主单片机程序

main.c

#include <stc12c5a60s2.h>
#include "Uart.h"
#include "Timer0.h"
#include "Key.h"
//#include "Digitron.h"
#include "stdio.h"
#define uchar unsigned char//自定义无符号字符型为uchar
#define uint unsigned int//自定义无符号整数型为uint
sbit LED = P1^6;//声明单片机P1.6端口为LED接口 
 void PortModeSet()//端口模式设置函数
{
   
  P0M1 = 0x00;
  P0M0 = 0x00;
  P1M1 = 0x00;
  P1M0 = 0x00;
  P2M1 = 0x00;
  P2M0 = 0x00;
  P3M1 = 0x00;
  P3M0 = 0x00;
  P4M1 = 0x00;
  P4M0 = 0x00;
 }
 void main()//主函数
{
   
//  uchar Data0 = 88;//定义数据变量为88
//  uint Data1 = 1250;//定义数据变量为1250
//  uint Data2 = 12.5;//定义数据变量为12.5
//  uint Data = 55;//定义数据变量为55
//  uchar ShiWeiData = Data / 10 + '0';//定义十位数值变量 取Data = 55 把数字55的十位数字5分解出来 注意分解出来的5是'5'表示字符5不是数字5 把字符5写入单片机数据缓存器 单片机数据缓存器会把字符5发送给串口调试助手软件接收区 如果串口调试助手软件接收区选十六进制数模式显示 那么字符5以ASCII码对应的十六进制数模式给显示出0x35来 如果串口调试助手软件接收区选文本模式显示 那么字符5以文本模式给显示出数字5来 为什么数据分解后+'0'? 因为单片机运算字符是字符对应的ASCII码的二进制数值或十进制数值或十六进制数值 这里取单片机运算字符对应的ASCII码十进制数值 如:字符'0'的ASCII码的十进制数是48 字符'1'的ASCII码的十进制数是49 字符'2'的ASCII码的十进制数是50 .... 字符'9'的ASCII码的十进制数是57 而编译器对于'0'会自动视为是'0'ASCII码的十进制数48 举例:拿Data = 251来分解 则有:Data/100=2 Data/10%10=5 data%10=1 要把数据型251转化成字符串为"251" 可拆开看成'2' '5' '1' 它们对应的ASCII码十进制数分别为50 53 49 而Data/100=2+'0'=2+48=50 Data/10%10=5+'0'=5+48=53 data%10=1+'0'=1+48=49 算出50 53 49这三个十进制数分别对上字符'2' '5' '1'的ASCII码十进制数为50 53 49 这就是为什么数据分解后+'0'的原因  
//  uchar GeWeiData = Data % 10 +'0';//定义个位数值变量 取Data = 55 把数字55的个位数字5分解出来 注意分解出来的5是'5'表示字符5不是数字5 把字符5写入单片机数据缓存器 单片机数据缓存器会把字符5发送给串口调试助手软件接收区 如果串口调试助手软件接收区选十六进制数模式显示 那么字符5以ASCII码对应的十六进制数模式给显示出0x35来 如果串口调试助手软件接收区选文本模式显示 那么字符5以文本模式给显示出数字5来 为什么数据分解后+'0'? 解释同上
  PortModeSet();//端口模式设置函数
  Uart1Init();//串行口1工作模式1的8位串行口波特率可变初始化函数 波特率为9600bps 晶振为12MHz
  Timer0Init();//定时器0的16位定时模式1用12分频定时2ms初始化函数 晶振为12MHz
//  DigitronBootDisplay();//数码管开机显示函数
//	printf("Wait For Serial Communication Test Start....\r\n");//打印提示语句
//  printf("Please Send a Frame Of Data....\r\n");//打印提示语句
  while(1)//主循环
 {
    
   KeyScanResult();//按键扫描结果函数
//   SendData('8');//注意'8'表示字符8不是数字8 把字符8写入单片机数据缓存器 单片机数据缓存器会字符8发送给串口调试助手软件接收区 如果串口调试助手软件接收区选十六进制数模式显示 那么字符8以ASCII码对应的十六进制数模式给显示出0x38来 如果串口调试助手软件接收区选文本模式显示 那么字符8以文本模式给显示出数字8来
//   SendData('\r');//回车 ASCII码对应的十六进制数为0x0a
//   SendData('\n');//换行 ASCII码对应的十六进制数为0x0d
//   SendString("Hello World!\r\n");//发送字符串函数
//   printf("Hello World!\r\n");//打印字符串
//   printf("Data0 = %bu\r\n",Data0);//打印无符号字符型数据
//   printf("Data1 = %u\r\n",Data1);//打印无符号整数型数据
//   printf("Data2 = %f\r\n",Data2);//打印单精度浮点型数据
//   SendData('a');//把字符a写入单片机数据缓存器 单片机数据缓存器会字符a发送给串口调试助手软件接收区 如果串口调试助手软件接收区选十六进制数模式显示 那么字符a以ASCII码对应的十六进制数模式给显示出0x61来 如果串口调试助手软件接收区选文本模式显示 那么字符a以文本模式给显示出字符a来  
//   SendData('1');//注意'1'表示字符1不是数字1 把字符1写入单片机数据缓存器 单片机数据缓存器会字符1发送给串口调试助手软件接收区 如果串口调试助手软件接收区选十六进制数模式显示 那么字符1以ASCII码对应的十六进制数模式给显示出0x31来 如果串口调试助手软件接收区选文本模式显示 那么字符1以文本模式给显示出数字1来  
//   SendData(ShiWeiData);//发送数据Data的十位数
//   SendData('\r');//换行
//   SendData('\n');//回车
//   SendData(GeWeiData);//发送数据Data的个位数
   if(ReceiveOverDataFlag == 1)//判断接收完数据标志位变量是否置1 即主单片机串行口数据缓存器已经接收完从单片机发送来的数据
  {
   
    ReceiveOverDataFlag = 0;//接收完数据标志位变量清0
	LED = 0;//点亮LED灯
	KeyPressNumber = ReceiveData;//接收数据变量含有的数值赋给按键按下数值变量 即主单片机串行口数据缓存器接收从单片机发送来的数据赋给按键按下数值变量
//    RS485DIR = 1;//启动RS485发送数据控制
//    ReceiveData = ReceiveData + 1;//接收数据变量累加
//    SendData(ReceiveData);//单片机通过串行口通信把来自计算机串口调试助手软件从发送区发送来的数据发回给计算机串口调试助手软件接收区显示出来
//    RS485DIR = 0;//启动RS485接收数据控制 准备下一次收发数据循环
//    printf("\r\n");//打印换行回车
//    printf("%bd\r\n",ReceiveData);//打印整数数据
   }
  }
 }

Uart.c

/*****关于8051系列单片机定时器溢出率、波特率和定时器初值(定时计数初值)之间计算的知识点*****/
/****
一、定时器溢出率计算公式
1、定时器溢出率:定时器每秒溢出的次数
2、定时器溢出率计算公式表
    定时方式           分频方式                      公式
方式1:16位定时器  12分频(即12T 默认值)  Ft=晶振频率/12/(65536-定时器初值)
方式2:8位定时器   12分频(即12T 默认值)  Ft=晶振频率/12/(256-定时器初值)
方式1:16位定时器      1分频(即1T)      Ft=晶振频率/1/(65536-定时器初值)
方式2:8位定时器       1分频(即1T)      Ft=晶振频率/1/(256-定时器初值)
二、波特率计算公式
1、波特率:每秒传输二进制位数的多少
2、波特率计算公式表
    定时方式             分频方式                        公式
方式1:16位定时器T1  12分频(即12T 默认值)  波特率=晶振频率/12/(65536-定时器初值)/4
方式2:8位定时器T1   12分频(即12T 默认值)  波特率=晶振频率/12/(256-定时器初值)*2^SMOD/32
方式1:16位定时器T2  12分频(即12T 默认值)  波特率=晶振频率/12/(65536-定时器初值)/4
方式1:16位定时器T1      1分频(即1T)      波特率=晶振频率/1/(65536-定时器初值)/4
方式2:8位定时器T1       1分频(即1T)      波特率=晶振频率/1/(256-定时器初值)*2^SMOD/32
方式1:16位定时器T2      1分频(即1T)      波特率=晶振频率/1/(65536-定时器初值)/4
三、根据波特率计算定时器初值(定时器定时计数)
    定时方式             分频方式                        公式
方式1:16位定时器T1  12分频(即12T 默认值)  定时器初值(定时计数)=65536-晶振频率/(48*波特率)
方式2:8位定时器T1   12分频(即12T 默认值)  定时器初值(定时计数)=256-晶振频率*2^SMOD/(384*波特率)
方式1:16位定时器T2  12分频(即12T 默认值)  定时器初值(定时计数)=65536-晶振频率/(48*波特率)
方式1:16位定时器T1      1分频(即1T)      定时器初值(定时计数)=65536-晶振频率/(4*波特率)
方式2:8位定时器T1       1分频(即1T)      定时器初值(定时计数)=256-晶振频率*2^SMOD/(32*波特率)
方式1:16位定时器T2      1分频(即1T)      定时器初值(定时计数)=65536-晶振频率/(4*波特率)
*****/
#include "Uart.h"
#include "stdio.h"
#define	uchar unsigned char	//定义无符号字符
#define	uint  unsigned int	//定义无符号整形
bit ReceiveOverDataFlag = 0;//定义接收完数据标志位变量为0
uint ReceiveData = 0;//定义接收数据变量为0
 void Uart1Init()//串行口1工作模式1的8位串行口波特率可变初始化函数 波特率为9600bps 晶振为12MHz
{
   
  SCON = 0x50;//工作模式1的8位串行口波特率可变
  AUXR &= 0xBF;//定时器时钟12T模式
  AUXR &= 0xFE;//串口1选择定时器1为波特率发生器
  PCON &= 0x7f;//波特率不加倍
  TMOD &= 0x0f;//定时器/计数器工作模式清0
  TMOD |= 0x20;//设定定时器/计数器为定时器 工作模式为8位自动重装定时器1模式2
  TH1 = 0xfd;//设定定时器1高八位初值
  TL1 = 0xfd;//设定定时器1低八位初值
  ET1 = 0;//禁止定时器1中断
  ES = 1;//允许串行口1中断
  EA = 1;//开总中断
  TR1 = 1;//打开定时器1
//  RS485DIR = 0;//启动RS485接收数据控制
 }
 void MasterSendData(uint Data)//主单片机发送数据函数 即主单片机给从单片机发送数据
{
   
  SBUF = Data;//把数据变量Data含有数据写入主单片机数据缓存器 主单片机数据缓存器会把数据变量Data含有数据发送给从单片机数据缓存器
  while(!TI)//当数据发送结束标志位变量TI为0 表示数据还没发送完 若数据发送结束标志位变量TI为1 表示数据已发送完 从而触发串口中断 最后需在串口中断程序中或者在其他程序中把数据发送结束标志位变量TI清0 才能进行下一次发送
  
  • 29
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值