通信按照传统的理解就是信息的傳输与交换UART(Universal Asynchronous Receiver/Transmitter,即通用异步收发器)串行通信是单片机串口初始化函数最常用的一种通信技术通常用于单片机串口初始化函数和电脑の间以及单片机串口初始化函数和单片机串口初始化函数之间的通信。
以下我们以STC98C52单片机串口初始化函数为例子简单讲述串行通信。
通信按照基本类型可以分为并行通信和串行通信并行通信时数据的各个位同时传送,可以实现字节为单位通信但是因为通信线多占用资源多,成本高比如使用STC89C52的P0口设置为P0 = 0xfe;一次给P0的8个IO口分别赋值,同时进行信号输出类似于有8个车道同时可以过去8辆车一样,这种形式就是並行的我们习惯上还称P0、P1、P2和P3为51单片机串口初始化函数的4组并行总线。
而串行通信就如同一条车道,一次只能一辆车过去如果一个0xfe這样一个字节的数据要传输过去的话,假如低位在前高位在后那发送方式就是0-1-1-1-1-1-1-1-1,一位一位的发送出去的要发送8次才能发送完一个字节。
在我们的STC89C52上有两个引脚,是专门用来做UART串口通信的一个是P3.0一个是P3.1,还分别有另外的名字叫做RXD和TXD这两个引脚是专门用来进行UART通信的,如果我们两个单片机串口初始化函数进行UART串口通信的话那基本的演示图如图1-1所示。
图中GND表示单片机串口初始化函数系统电源的参考哋,TXD是串行发送引脚RXD是串行接收引脚。两个单片机串口初始化函数之间要通信首先电源基准得一样,所以我们要把两个单片机串口初始化函数的GND相互连起来然后单片机串口初始化函数1的TXD引脚接到单片机串口初始化函数2的RXD引脚上,即此路为单片机串口初始化函数1发送而單片机串口初始化函数2接收的通道单片机串口初始化函数1的RXD引脚接到单片机串口初始化函数2的TXD引脚上,即此路为单片机串口初始化函数2發送而单片机串口初始化函数2接收的通道这个示意图就体现了两个单片机串口初始化函数各自收发信息的过程。
当单片机串口初始化函數1想给单片机串口初始化函数2发送数据时比如发送一个0xE4这个数据,用二进制形式表示就是0b在UART通信过程中,是低位先发高位后发的原則,那么就让TXD首先拉低电平持续一段时间,发送一位0然后继续拉低,再持续一段时间又发送了一位0,然后拉高电平持续一段时间,发了一位1......一直到把8位二进制数字0b全部发送完毕这里就牵扯到了一个问题,就是持续的这“一段时间”到底是多久从这里引入我们通信中的另外重要概念——波特率,也叫做比特率
波特率就是发送一位二进制数据的速率,习惯上用baud表示即我们发送一位数据的持续时間=1/baud。在通信之前单片机串口初始化函数1和单片机串口初始化函数2首先都要明确的约定好他们之间的通信波特率,必须保持一致收发双方才能正常实现通信,这一点大家一定要记清楚
约定好速度后,我们还要考虑第二个问题数据什么时候是起始,什么时候是结束呢鈈管是提前接收还是延迟接收,数据都会接收错误在UART串行通信的时候,一个字节是8位规定当没有通信信号发生时,通信线路保持高电岼当要发送数据之前,先发一位0表示起始位然后发送8位数据位,数据位是先低后高的顺序数据位发完后再发一位1表示停止位。这样夲来要发送一个字节8位数据而实际上我们一共发送了10位,多出来的两位其中一位起始位一位停止位。而接收方呢原本一直保持的高電平,一旦检测到来了一位低电平那就知道了要开始准备接收数据了,接收到8位数据位后然后检测到停止位,再准备下一个数据的接收了我们图示看一下,如图1-2所示
如图1-2串口数据发送示意图,实际上是一个时域示意图就是信号随着时间变化的对应关系。比如在单爿机串口初始化函数的发送引脚上左边的是先发生的,右边的是后发生的数据位的切换时间就是波特率分之一秒,如果能够理解时域嘚概念后边很多通信的时序图就很容易理解了。
随着技术的发展工业上还有RS232串口通信的大量使用,但是商业技术的应用上已经慢慢嘚使用USB转UART技术取代了RS232串口,绝大多数笔记本电脑已经没有串口这个东西了那我们要实现单片机串口初始化函数和电脑之间的通信该如何辦呢?
我们只需要在我们电路上添加一个USB转串口芯片就可以成功实现USB通信协议和标准UART串行通信协议的转换,在我们的开发板上我们使鼡的是CH340G这个芯片,如图1-3所示
CH340G这个电路很简单,把电源电路晶振电路接好后,6脚和7脚的UD+和UD-分别接USB口的2个数据引脚上去3脚和4脚接到了我們单片机串口初始化函数的TXD和RXD上去。
CH340G的电路里2脚位置加了个4148的二极管是一个小技巧。因为我们的STC89C52RC这个单片机串口初始化函数下载程序需偠冷启动就是先点下载后上电,上电瞬间单片机串口初始化函数会先检测需要不需要下载程序虽然单片机串口初始化函数的VCC是由开关來控制,但是由于CH340G的2脚是输出引脚如果没有此二极管,开关后级单片机串口初始化函数在断电的情况下CH340G的2脚和单片机串口初始化函数嘚P3.0(即RXD)引脚连在一起,有电流会通过这个引脚流入后级电路并且给后级的电容充电造成后级有一定幅度的电压,这个电压值虽然只有兩三伏左右但是可能会影响到我们的冷启动。加了二极管后一方面不影响通信,另外一个方面还可以消除这种问题这个地方可以暂時作为了解,大家如果自己做这块电路可以参考一下。
为了让大家充分理解UART串口通信的原理我们先用P3.0和P3.1这两个当做IO口来进行模拟实际串口通信的过程,原理搞懂后我们再使用寄存器配置实现串口通信过程。
对于UART串口波特率常用的值是1200、2400、4800、9600、14400、19200、28800、38400、57600、115200、128000、256000等速率。IO口模拟UART串行通信程序是一个简单的演示程序我们使用串口调试助手下发一个数据,数据加1后再自动返回。波特率是我们程序设定好嘚选择我们程序中让一个数据位持续时间是1/9600秒,那这个地方选择波特率就是选9600校验位选N,数据位8停止位1。
串口调试助手的实质就是峩们利用电脑上的UART通信接口通过这个UART接口发送数据给我们的单片机串口初始化函数,也可以把我们的单片机串口初始化函数发送的数据接收到这个调试助手界面上
因为初次接触通信方面的技术,所以我对这个程序进行一下解释大家可以边看我的解释边看程序,把底层原理先彻底弄懂
变量定义部分就不用说了,直接看main主函数首先是对通信的波特率的设定,在这里我们配置的波特率是9600那么串口调试助手也得是9600。配置波特率的时候我们用的是定时器0的模式2。模式2中不再是TH0代表高8位,TL0代表低8位了而只有TL0在进行计数了。当TL0溢出后鈈仅仅会让TF0变1,而且还会将TH0中的内容重新自动装到TL0中这样有一个好处,我们可以把我们想要的定时器初值提前存在TH0中当TL0溢出后,TH0自动紦初值就重新送入TL0了全自动的,不需要程序上再给TL0重新赋值了配置方式很简单,大家可以自己看下程序并且计算一下初值
波特率设置好以后,打开中断然后等待接收串口调试助手下发的数据。接收数据的时候首先要进行低电平检测 while (PIN_RXD),若没有低电平则说明没有数据一旦检测到低电平,就进入启动接收函数StartRXD()接收函数最开始启动半个波特率周期,初学可能这里不是很明白大家回头看一下我们的图1-2裏边的串口数据示意图,信号在数据位电平变化的时候去读因为时序上的误差以及信号稳定性的问题很容易读错数据,所以我们希望在信号最稳定的时候去读数据除了信号变化的那个沿的位置外,其他位置都很稳定那么我们现在就约定在信号中间位置去读取电平状态,这样能够保证我们信号读的是对的
一旦读到了起始信号,我们就把当前状态设定成接受状态并且打开定时器中断,第一次是半个周期进入中断后对起始位进行二次判断一下,确认一下起始位是低电平而不是一个干扰信号。以后每经过9600分之一秒进入一次中断并且紦这个引脚的状态读到RxdBuf里边。等待接收完毕之后我们再把这个RxdBuf加1,再通过TXD引脚发送出去同样需要先发一位起始位,然后发8个数据位洅发结束位,发送完毕后程序运行到while (PIN_RXD),等待第二轮信号接收的开始
if (!PIN_RXD) //起始位为0时,清零接收缓冲器准备接收数据位 else //起始位不为0时,中圵接收 if (PIN_RXD) //接收脚为1时缓冲器最高位置1;为0时不处理即仍保持移位后的0