基于STM32F103入门4——串口通信

发布者:忠正最新更新时间:2022-09-23 来源: csdn关键字:STM32F103  入门  串口通信 手机看文章 扫描二维码
随时随地手机看文章

最近能抽点时间学一下stm32了,串口通信用的挺多的,比如wifi模块,GSM模块,指纹模块等等…在这里用自己理解的写一下总结,如果有误的话请大家多多指点。

串口的基本概念

串行和并行

串行

串行是一位一位的传输。

常用的有 USART、IIC、SPI等…

串行也分为 同步通信 和 异步通信

同步:

就是一般有一根时钟线,有时钟就可以大家一起同步嘛。靠时钟来约定。一根数据线。一般一个时钟传输一个Bit位。同步的话 他们大部分都是有效数据来的。但是对于双方的时钟允许误差较小。


异步:

异步就是我们平常玩串口用的最多的,它不像同步有个时钟,异步是没有时钟,那我们得为了数据不出错,所以通过 起始位、奇偶校验位、停止位这些来降低数据的错误。

所以异步的有效数据就没有同步的有效数据那么多。效率也没有同步的那么高。


优点:传输距离远、抗干扰能力强、成本较低

缺点:传输速率慢


并行

并行是指多比特数据同时通过并行线进行传送,这样数据传送速度大大提高。

简单的理解就是 比如我们用的LCD1602不是有一共10几个引脚么,然后8根是数据线,这就是属于并行,它们通常需要数据总线(八、十六或更多线路)。


优点:线比较多嘛,所以我传输就快,这个容易理解。

缺点:线这么多,那成本也会高嘛,别人一根线搞定了,你用了8根是吧。然后距离长了,那也容易受到干扰。


通信方式

全双工

就是双方都可以同时发送和接收,相当于我们的打电话。


半双工

在同一时刻,只能一个发送、一个接收,相当于我们的对讲机。


单工

这个就是一个只能发,一个只能收。相当于广播,他那边只能发,我们只能听。


波特率

首先讲一下比特率

比特率:每秒钟传输的二进制位数,单位是(bit/s)

跟波特率有点区别的,但是有时候也一样。


波特率:表示每秒钟传输的码元个数。

码元:比如我们玩51的时候基本都是5V OV那么 5V相当于二进制1 0V就相当于二进制0。

5V——1

0V——0


有时候多个,比如

0V——00

2V——01

4V——10

6V——11

这样就跟比特率不一样了。


波特率计算

波特率 = Fck/(16*USARTDIV)

Fck:串口的时钟

USARTDIV:无符号定点数


比如我想设置115200波特率

时钟是72M

那USARTDIV = 39.0625

在这里插入图片描述

串口的配置

代码编写步骤

我这里用USART1来举例。我这里是A9(TX)、A10(RX)(异步通信)

1:打开对应 的时钟

1.1:GPIO的时钟

1.2:串口的时钟


2.配置GPIO结构体

2.1:引脚

2.2:模式(输入还是输出)

2.3:速率(输出才用 输入不用)

2.4:对结构体成员初始化


3配置串口结构体

3.1:波特率

3.2:有效数据位

3.3:停止位

3.4:奇偶校验位

3.5:硬件控制流

3.6:模式

3.7:对结构体成员初始化


4配置NVIC中断优先级(misc.h)

如果不需要中断可以不配置NVIC

4.1:选择哪个组 NVIC_PriorityGroupConfig

4.2::哪个中断通道 (stm32f10x.h里 IRQn_Type结构体里找)

4.3:主优先级

4.4:次优先级

4.5:通道使能开启

4.6:对成员初始化

4.7:串口中断配置 USART_ITConfig(采用什么方式中断)


5串口使能

USART_Cmd();


6:编写中断服务函数

函数名字在启动文件里找。


串口的重定向

比如我们常用输出函数

printf();

putchar();

常用的输入函数

scanf();

getchar();


输出

==记得包含头文件 stdio.h ==

其次你添加下面的函数既可以用printf 和 putchar了。


/*重定向C库函数printf到串口*/

int fputc(int ch, FILE *f)

{

USART_SendData(USART1, (uint8_t)ch);; //发送一字节到串口

while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET); //等待发送寄存器为空 证明发送完

return ch;

}


输入

==记得包含头文件 stdio.h ==

其次你添加下面的函数既可以用scanf 和 getchar了。


/*重定向C库函数 scanf到串口*/

int fgetc(FILE *f)

{

while(USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == RESET); //如果读数据寄存器非空

return (int)USART_ReceiveData(USART1);

}


串口的例程

我这里就实现一下 电脑端串口助手发送单字符到单片机,单片机接收到就立刻也发给电脑串口助手。

比如 我在串口助手发送了 A

然后单片机接收到之后,让单片机发送receive data:A


main.c

#include "stm32f10x.h"

#include "usart.h"


int main()

{

usart1_init(115200); //串口1初始化函数

while(1)

{

}

}


/*因为用的是中断接收所以少不了中断服务函数*/

/*串口1 中断服务函数*/

void USART1_IRQHandler(void)

{

uint8_t data;

if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //再次判断是否中断是否发生

{

data = USART_ReceiveData(USART1);

}

usart_sendString(USART1,"receive data:");

usart_sendByte(USART1,data);

usart_sendByte(USART1,'n');

}


usart.c

/* 配置串口1 优先级 函数 */

static void NVIC_USART1_configuration(void)

{

NVIC_InitTypeDef NVIC_initStruction;

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0); //组

NVIC_initStruction.NVIC_IRQChannel = USART1_IRQn; //串口1中断

NVIC_initStruction.NVIC_IRQChannelPreemptionPriority = 0;   //主优先级

NVIC_initStruction.NVIC_IRQChannelSubPriority = 0; //次优先级

NVIC_initStruction.NVIC_IRQChannelCmd = ENABLE; //使能

NVIC_Init(&NVIC_initStruction);

}


/* 配置串口1 函数*/

void usart1_init(uint32_t baudRate)

{

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //打开GPIOA时钟

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); //打开串口1时钟

GPIO_InitTypeDef GPIO_initStruction;

USART_InitTypeDef USART_initStruction;


/*配置GPIOA  TX */

GPIO_initStruction.GPIO_Pin = USART1_TX; // TX

GPIO_initStruction.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出

GPIO_initStruction.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init(GPIOA, &GPIO_initStruction);

/*配置GPIOA RX */

GPIO_initStruction.GPIO_Pin = USART1_RX; // RX

GPIO_initStruction.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入

GPIO_Init(GPIOA, &GPIO_initStruction);

/*配置USART1 TX和RX */

USART_initStruction.USART_BaudRate = baudRate; //波特率

USART_initStruction.USART_WordLength = USART_WordLength_8b; //8位有效数据位

USART_initStruction.USART_StopBits = USART_StopBits_1; //1个停止位

USART_initStruction.USART_Parity = USART_Parity_No; //无奇偶校验位

USART_initStruction.USART_HardwareFlowControl = USART_HardwareFlowControl_None; //不硬件控制流

USART_initStruction.USART_Mode = USART_Mode_Tx | USART_Mode_Rx; //发送 和 接收

USART_Init(USART1, &USART_initStruction);

NVIC_USART1_configuration(); //串口1中断优先级配置

USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//使能接收中断

USART_Cmd(USART1, ENABLE); //使能串口1

}


效果呈现

在这里插入图片描述

关键字:STM32F103  入门  串口通信 引用地址:基于STM32F103入门4——串口通信

上一篇:基于STM32F103入门3——外部中断
下一篇:基于STM32F103入门2——按键点灯

小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved