当前位置: 首页 > news >正文

单片机-89C51部分:9、串行口通讯

飞书文档https://x509p6c8to.feishu.cn/wiki/WSh3wnADkixHspk7kc8c5esRnad

一、什么是串口?它的作用?

串行口,简称为串口,什么是串口?它的作用是什么?

两个人交流,一般通过在说话在空气中产生的声波传输,两台机器交流,我们可以模拟摩斯密码一样,其中一台机器通过IO发送对应频率高低电平的脉冲,另一台机器接收进行解析,但这种操作IO的方式需要我们考虑的问题很多,多快的频率,谁发谁收,丢失了一个脉冲怎么办等等,于是后面就有了规范这些交流的协议,简称为通讯协议,基于通讯方式的不同,也有了串行通讯和并行通讯两种通讯方式。

串口是一种应用十分广泛的通讯接口,串口成本低、容易使用、通信线路简单,可实现两个设备的互相通信。

串行通信

串行通信是指使用一条数据线,将数据一位一位地依次传输,每一位数据占据一个固定的时间长度。其只需要少数几条线就可以在系统间交换信息,特别适用于计算机与计算机,计算机与外设之间的远距离通信,先传输低位在传输高位。如下图所示:

并行通信

并行通信通常是将数据字节的各位用多条数据线同时进行传送,通常是8位,16位,32位等数据一起传输。如下图所示:

对比:

串行通信的特点:传输线少,长距离传送时成本低,且可以利用电话网等现成的设备,但数据的传送控制比并行通信复杂。

并行通信的特点:控制简单,传输速度快;由于传输线较多,长距离传送时成本高且接收方的各位同时接受存在困难,抗干扰能力差。

现阶段绝大部分的通讯口都使用串口。

二、串口的参数

串行通信的基本方式

单工通信:数据只能单方向传输。

半双工通信:通信双方交替进行双向数据传输,但两个方向的传输不能同时进行。

全双工通信:通信双方可同时进行数据收发的工作方式。51单片机的串行口是全双工传输方式。

串口电平标准

是传输线缆中人为规定的电压与数据的对应关系,串口常用的电平标准有如下三种:

• TTL电平(transistor transistor logic ): +3V~+5V 表示 1 , 0V 表示 0

• RS232 电平: -3~-15V 表示 1 , +3~+15V 表示 0

• RS485 电平:两线压差 +2~+6V 表示 1 , -2~-6V 表示 0 (差分信号)

串口数据结构

一个完整的串行数据,也就是一个数据帧(Data frame),包括起始位、数据位、停止位、奇偶校验位。数据位前后即帧头和帧尾,包含一些必要的控制信息。其中,MSB(Most Significant Bit)是指低地址存放最高有效字节,LSB(Least Significant Bit)则是低地址存放最低有效字节。

  1. 20 - > 0x14 -> 0b0001 0100
  2. MSB:     0010 1000
  3. LSB:      0001 0100

串口波特率

通俗解析就是,波特率越高,传输速度越快。

常见的串口典型的“波特率”值是 300/1200/2400/9600/19200/38400/115200 /230400等。

串口通信速率,单位时间内传输二进制的位数(例波特率为9600,指1s内传输9600位,则传输一位需要1/9600=104.17us)

串口校验位

N 无校验

不加校验位,可以少传输一位数据

O 奇校验

a要传输的数据中(不包含校验位)有奇数个‘1’ 则校验位为‘0’, 反之为‘1’

例: 数据‘1111 000’ 偶数个‘1’ 所以添加校验位为‘1’ 整体为‘1111 0000 1’

E  偶校验

要传输的数据中(不包含校验位)有偶数个‘1’ 则校验位为‘0’, 反之为‘1’

M(Mark 标记、符合)检验位固定为1

S(Space 空间、空地)校验位固定为0

串口停止位

停止位,停止位是一帧数据结束的标志,可以是1bit、1.5bit或者2bit逻辑“1” 高电平

如果没有停止位,接收设备就无法知道何时一个数据包结束,从而无法正确地处理接收到的数据。

空闲位

空闲位不算是串口报文内的数据, 它是发送完一组报文后,总线会自动将电平拉高,产生1bit 逻辑“1”的空闲位

串口调试助手

https://alithon.com/downloads

主要使用CH340驱动软件和逻辑分析仪驱动软件,可以点击飞书文档链接下载

USB转串口助手:

逻辑分析仪:

三、51的串口使用

硬件接线

串口相关寄存器:

一般常用的是SCON SBUF PCON IE,如果需要配置不同中断优先级的则需要配置IPH IP,SADEN、SADDR用于多机通讯,比较少用。

模式选择

STC89C52 有 1 个 UART, 有四种工作模式:

模式0:同步移位寄存器,主要用于扩展并行输入或输出口。

模式18UART,波特率可变(常用)

模式2:9位UART,波特率固定(多出的1位为校验位)

模式3:9位UART,波特率可变(多出的1位为校验位)

这里简单说明下常用和其它的区别,常用的就是你使用串口功能时,一般使用模式1就能满足所有需求,像我工作这么久,做了非常多的项目,其它模式基本上没用过,所以学习时把常用的学会,原理搞懂就可以了,后面根据项目需求有目的性学习,而不是把所有细节都学一遍。

模式选择寄存器

SM0 = 0;
SM1 = 1;


SCON =
0x40;                //0100 0000 串口工作模式1

模式1说明

发送数据相关寄存器配置

发送流程

SBUF 串口数据缓冲寄存器:可写入需要发送的数据

时钟源选择,影响波特率

Timer1 Overflow:定时器1溢出作为时钟源,用于串口通讯的波特率

查表法:

方式一:
TMOD = 0X20;   //设置定时器1工作模式2 8位自动重载
TH1 = 0xFD;    //设定初值
TL1 = 0XFD;    //设定装载值
TR1 = 1;       //打开计数器


TMOD &= 0x0F;  //清空TMOD中定时器1相关
TMOD |= 0X20;  //设置定时器1工作模式2 8位自动重载,这里为什么使用|=而不使用=呢,为了避免清除定时器0配置
TH1 = 0xFD;    //设定初值
TL1 = 0XFD;    //设定装载值
TR1 = 1;       //打开计数器

公式法:


 

如果需要设置波特率为9600=(1/32)*(11059000/12/(256-TH1))

SMOD:波特率加倍选择位,复位为0,可以不设置

方式一:
TMOD &= 0x0F;  //清空TMOD中定时器1相关
TMOD |= 0X20;  //设置定时器1工作模式2 8位自动重载,这里为什么使用|=而不使用=呢,为了避免清除定时器0配置
TH1 = 0xFD;    //设定初值
TL1 = 0XFD;    //设定装载值
TR1 = 1;       //打开计数器
PCON |= 0X00;  //设置B7为0

校验位设置

TB8:校验位,在串口模式23作为校验位,模式18bit的,无需校验位

发送数据

发送字节函数:

#include <reg52.h>void UartInit()                //9600bps@11.0592MHz
{SCON = 0x40;        //0100 0000 串口工作模式1TMOD &= 0x0F;       //清空TMOD中定时器1相关TMOD |= 0X20;       //设置定时器1工作模式2:8位自动重载TH1 = 0xFD;         //设定定时初值TL1 = 0XFD;TR1 = 1;            //启动定时器1
}//主函数
void main()
{UartInit();//调用串口初始化函数SBUF=0x30;while(1){}       
}

发送字符串函数:

这里还需要注意一个点是发送完成最后一位后,会置TI=1,所以我们可以用TI寄存器来判断是否发送完。

#include <reg52.h>void UartInit()             //9600bps@11.0592MHz
{SCON = 0x40;        //0100 0000 串口工作模式1TMOD &= 0x0F;       //清空TMOD中定时器1相关TMOD |= 0X20;       //设置定时器1工作模式2:8位自动重载TH1 = 0xFD;         //设定定时初值TL1 = 0XFD;TR1 = 1;            //启动定时器1
}void send_string(unsigned char str[])
{unsigned char i=0;while(str[i]!='\0')//判断是否到字符串尾{SBUF = str[i];while(TI==0);    //等待发送完成,发送完成TI会置1TI=0;            //下次发送前,要手动将TI置0i++;             //下次发送}       
}//主函数
void main()
{UartInit();//调用串口初始化函数send_string("hello world!!");while(1){}       
}

串口接收数据相关寄存器配置

波特率设置

波特率部分和发送是一样,于是初始化函数就变为

void UartInit()                //9600bps@11.0592MHz
{SCON = 0x40;            //0101 0000 串口工作模式1TMOD &= 0x0F;           //清空TMOD中定时器1相关TMOD |= 0X20;           //设置定时器1工作模式2:8位自动重载TH1 = 0xFD;             //设定定时初值TL1 = 0XFD;TR1 = 1;                //启动定时器1
}

允许接收寄存器设置

REN 禁止/允许串口接收控制位,为1时,才能接收。

SM0 = 0;
SM1 = 1;
REN = 1;//
允许接收数据

SCON =
0x50;                //0101 0000 串口工作模式1 允许接收

接收有效数据标志位

RI:接收有效数据标志位

串口中断

接收到有效数据时,会触发串口中断,这时候,我们开启串口中断相关的寄存器,就可以在中断服务函数中处理相关逻辑,不需要在大循环中一直判断啦。


#include <reg52.h>
sbit led=P2^7;//串口初始化函数
void UartInit()                //9600bps@11.0592MHz
{SCON = 0x50;                //0101 0000 串口工作模式1TMOD &= 0x0F;                //清空TMOD中定时器1相关TMOD |= 0X20;       //设置定时器1工作模式2:8位自动重载TH1 = 0xFD;                //设定定时初值TL1 = 0XFD;TR1 = 1;                //启动定时器1ES=1;//打开串行通信中断EA=1;//打开总中断
}
//串行中断函数
void Uart_receive() interrupt 4
{if(RI==1)//RI=1说明串口接收到了数据{                char receive;RI=0;//RI置0保证下次接收receive=SBUF;//将从串口接收到的数据报存到变量中//判断接收的数据,作出相应的操作if(receive=='O')led=0;if(receive=='C')led=1;       }
}
//主函数
void main()
{UartInit();while(1){}
}

Printf发送数据

#include <reg52.h>
#include <stdio.h> void UartInit(void)                //9600bps@11.0592MHz
{PCON &= 0x7F;                //波特率不倍速SCON = 0x50;                //8位数据,可变波特率TMOD &= 0x0F;                //清除定时器1模式位TMOD |= 0x20;                //设定定时器1为8位自动重装方式TL1 = 0xFD;                //设定定时初值TH1 = 0xFD;                //设定定时器重装值ET1 = 0;                //禁止定时器1中断TR1 = 1;                //启动定时器1
}/*
**重写printf调用的putchar函数,重定向到串口输出
**需要引入头文件<stdio.h>
*****/
char putchar(char dat){//输出重定向到串口SBUF = dat;     //写入发送缓冲寄存器while(!TI);    //等待发送完成,TI发送溢出标志位 置1TI = 0;      //对溢出标志位清零return dat;  //返回给函数的调用者printf
}//主函数
void main()
{UartInit();//调用串口初始化函数printf("hello\r\n");while(1){}       
}

printf格式化输出

c语言中的格式输出稍微一些不同
例如打印 unsigned char 类型的数据需要使用 %bd

unsigned char dat1 = 48;
printf("char-->%bd\r\n",dat1);   //无符号字符型使用%bd显示十进制数

相关文章:

  • Gitea windows服务注册,服务启动、停止、重启脚本
  • MySQL慢查询日志分析方法
  • BLE技术,如何高效赋能IoT短距无线通信?
  • 应用安全系列之四十七:NoSQL注入
  • 14.外观模式:思考与解读
  • IoTDB数据库建模与资源优化指南
  • 从拒绝采样到强化学习,大语言模型推理极简新路径!
  • Tailwind CSS 实战:基于 Kooboo 构建企业官网页面(三)
  • Webshell管理工具的流量特征
  • Selenium 与 Playwright:浏览器自动化工具的深度对比
  • python jupyter notebook
  • 麒麟OS系统的Python程序和应用部署
  • 给 BBRv2/3 火上浇油的 drain-to-target
  • 使用DDR4控制器实现多通道数据读写(十)
  • Thinkphp开发自适应职业学生证书查询系统职业资格等级会员证书管理网站
  • 【PyTorch动态计算图原理精讲】从入门到灵活应用
  • react-native-vector-icons打包报错并且提示:copyReactNativeVectorIconFonts相关信息
  • 20_大模型微调和训练之-基于LLamaFactory+LoRA微调LLama3后格式合并
  • 详解大语言模型生态系统概念:lama,llama.cpp,HuggingFace 模型 ,GGUF,MLX,lm-studio,ollama这都是什么?
  • LeetCode 2302 统计得分小于K的子数组数目(滑动窗口)
  • 证监会:坚决拥护党中央对王建军进行纪律审查和监察调查的决定
  • 向总书记汇报具身智能发展的“稚辉君”:从期待到兴奋再到备受鼓舞
  • 宋徽宗《芙蓉锦鸡图》亮相,故宫首展历代动物绘画
  • 人社部:将制定提前领取个人养老金相关办法
  • 外交部亚洲司司长刘劲松向菲方严肃交涉
  • 呼伦贝尔市委常委、组织部长闫轶圣调任内蒙古交通集团党委副书记