关于stm32串口必须要学的5个串口以及串口应用和注意事项

发布者:Yuexin888最新更新时间:2025-01-20 来源: jianshu关键字:stm32  串口  注意事项 手机看文章 扫描二维码
随时随地手机看文章

 串口是我们常用的一个数据传输接口,STM32F103系列单片机共有5个串口。

  其中1-3是通用同步/异步串行接口USART(Universal Synchronous/Asynchronous Receiver/Transmitter)。

  4,、5是通用异步串行接口UART(Universal Asynchronous Receiver/Transmitter)。

  看完文章总结可以看下边的资料了解详细情况

  (stm32 USART串口应用)

http://www.makeru.com.cn/live/1392_1164.html?s=45051

  通过Z-stack协议栈实现串口透传

http://www.makeru.com.cn/live/1758_330.html?s=45051

  配置串口包括三部分内容:

  1. I/O口配置:TXD配置为复用推挽输出(GPIO_Mode_AF_PP),RXD配置为浮空输入(GPIO_Mode_IN_FLOATING);

  2. 串口配置:波特率等;

  3. 中断向量配置:一般用中断方式接收数据。

  注意事项:

  1. USART1是挂在APB2,使能时钟命令为:

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE );

  其他几个则挂在APB1上,如2口:

  RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE );

  2. 配置4口和5口的时候,中断名为UART4、UART5,中断入口分别为

  UART4_IRQn、UART5_IRQn

  对应的中断服务函数为

  void UART4_IRQHandler(void)

  和

  void UART5_IRQHandler(void)。

  下面是5个串口的配置函数和收发数据函数代码:

  #include 'stm32f10x.h'

  #include 'misc.h'

  #include 'stm32f10x_gpio.h'

  #include 'stm32f10x_usart.h'

  void USART1_Configuration(void)

  {

  GPIO_InitTypeDef GPIO_InitStructure;

  USART_InitTypeDef USART_InitStructure;

  NVIC_InitTypeDef NVIC_InitStructure;

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE );

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE );

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //USART1 TX;

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

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

  GPIO_Init(GPIOA, &GPIO_InitStructure); //端口A;

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; //USART1 RX;

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

  GPIO_Init(GPIOA, &GPIO_InitStructure); //端口A;

  USART_InitStructure.USART_BaudRate = 9600; //波特率;

  USART_InitStructure.USART_WordLength = USART_WordLength_8b; //数据位8位;

  USART_InitStructure.USART_StopBits = USART_StopBits_1; //停止位1位;

  USART_InitStructure.USART_Parity = USART_Parity_No ; //无校验位;

  USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;

  //无硬件流控;

  USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;

  //收发模式;

  USART_Init(USART1, &USART_InitStructure);//配置串口参数;

  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置中断组,4位抢占优先级,4位响应优先级;

  NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; //中断号;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //抢占优先级;

  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //响应优先级;

  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  NVIC_Init(&NVIC_InitStructure);

  USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);

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

  }

  void USART1_Send_Byte(u8 Data) //发送一个字节;

  {

  USART_SendData(USART1,Data);

  while( USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET );

  }

  void USART1_Send_String(u8 *Data) //发送字符串;

  {

  while(*Data)

  USART1_Send_Byte(*Data++);

  }

  void USART1_IRQHandler(void) //中断处理函数;

  {

  u8 res;

  if(USART_GetITStatus(USART1, USART_IT_RXNE) == SET) //判断是否发生中断;

  {

  USART_ClearFlag(USART1, USART_IT_RXNE); //清除标志位;

  res=USART_ReceiveData(USART1); //接收数据;

  USART1_Send_Byte(res); //用户自定义;

  }

  }

  void USART2_Configuration(void)

  {

  GPIO_InitTypeDef GPIO_InitStructure;

  USART_InitTypeDef USART_InitStructure;

  NVIC_InitTypeDef NVIC_InitStructure;

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE );

  RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE );

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; //USART2 TX;

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

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

  GPIO_Init(GPIOA, &GPIO_InitStructure); //端口A;

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3; //USART2 RX;

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

  GPIO_Init(GPIOA, &GPIO_InitStructure); //端口A;

  USART_InitStructure.USART_BaudRate = 9600; //波特率;

  USART_InitStructure.USART_WordLength = USART_WordLength_8b; //数据位8位;

  USART_InitStructure.USART_StopBits = USART_StopBits_1; //停止位1位;

  USART_InitStructure.USART_Parity = USART_Parity_No ; //无校验位;

  USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;

  //无硬件流控;

  USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;

  //收发模式;

  USART_Init(USART2, &USART_InitStructure);//配置串口参数;

  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置中断组,4位抢占优先级,4位响应优先级;

  NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn; //中断号;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //抢占优先级;

  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //响应优先级;

  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  NVIC_Init(&NVIC_InitStructure);

  USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);

  USART_Cmd(USART2, ENABLE); //使能串口;

  }

  void USART2_Send_Byte(u8 Data) //发送一个字节;

  {

  USART_SendData(USART2,Data);

  while( USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET );

  }

  void USART2_Send_String(u8 *Data) //发送字符串;

  {

  while(*Data)

  USART2_Send_Byte(*Data++);

  }

  void USART2_IRQHandler(void) //中断处理函数;

  {

  u8 res;

  if(USART_GetITStatus(USART2, USART_IT_RXNE) == SET) //判断是否发生中断;

  {

  USART_ClearFlag(USART2, USART_IT_RXNE); //清除标志位;

  res=USART_ReceiveData(USART2); //接收数据;

  USART2_Send_Byte(res); //用户自定义;

  }

  }

  void USART3_Configuration(void)

  {

  GPIO_InitTypeDef GPIO_InitStructure;

  USART_InitTypeDef USART_InitStructure;

  NVIC_InitTypeDef NVIC_InitStructure;

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE );

  RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3, ENABLE );

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; //USART3 TX;

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

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

  GPIO_Init(GPIOB, &GPIO_InitStructure); //端口B;

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11; //USART3 RX;

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

  GPIO_Init(GPIOB, &GPIO_InitStructure); //端口B;

  USART_InitStructure.USART_BaudRate = 9600; //波特率;

  USART_InitStructure.USART_WordLength = USART_WordLength_8b; //数据位8位;

  USART_InitStructure.USART_StopBits = USART_StopBits_1; //停止位1位;

  USART_InitStructure.USART_Parity = USART_Parity_No ; //无校验位;

  USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;

  //无硬件流控;

  USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;

  //收发模式;

  USART_Init(USART3, &USART_InitStructure);//配置串口参数;

  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置中断组,4位抢占优先级,4位响应优先级;

  NVIC_InitStructure.NVIC_IRQChannel = USART3_IRQn; //中断号;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //抢占优先级;

  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //响应优先级;

  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  NVIC_Init(&NVIC_InitStructure);

  USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);

  USART_Cmd(USART3, ENABLE); //使能串口;

  }

  void USART3_Send_Byte(u8 Data) //发送一个字节;

  {

  USART_SendData(USART3,Data);

  while( USART_GetFlagStatus(USART3, USART_FLAG_TC) == RESET );

  }

  void USART3_Send_String(u8 *Data) //发送字符串;

  {

  while(*Data)

  USART3_Send_Byte(*Data++);

  }

  void USART3_IRQHandler(void) //中断处理函数;

  {

  u8 res;

  if(USART_GetITStatus(USART3, USART_IT_RXNE) == SET) //判断是否发生中断;

  {

  USART_ClearFlag(USART3, USART_IT_RXNE); //清除标志位;

  res=USART_ReceiveData(USART3); //接收数据;

  USART3_Send_Byte(res); //用户自定义;

  }

  }

  void UART4_Configuration(void)

  {

  GPIO_InitTypeDef GPIO_InitStructure;

  USART_InitTypeDef USART_InitStructure;

  NVIC_InitTypeDef NVIC_InitStructure;

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE );

  RCC_APB1PeriphClockCmd(RCC_APB1Periph_UART4, ENABLE );

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; //UART4 TX;

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

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

  GPIO_Init(GPIOC, &GPIO_InitStructure); //端口C;

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11; //UART4 RX;

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

  GPIO_Init(GPIOC, &GPIO_InitStructure); //端口C;

  USART_InitStructure.USART_BaudRate = 9600; //波特率;

  USART_InitStructure.USART_WordLength = USART_WordLength_8b; //数据位8位;

  USART_InitStructure.USART_StopBits = USART_StopBits_1; //停止位1位;

  USART_InitStructure.USART_Parity = USART_Parity_No ; //无校验位;

  USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;

  //无硬件流控;

  USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;

  //收发模式;

  USART_Init(UART4, &USART_InitStructure);//配置串口参数;

  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置中断组,4位抢占优先级,4位响应优先级;

  NVIC_InitStructure.NVIC_IRQChannel = UART4_IRQn; //中断号;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //抢占优先级;

  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //响应优先级;

  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  NVIC_Init(&NVIC_InitStructure);

  USART_ITConfig(UART4, USART_IT_RXNE, ENABLE);

  USART_Cmd(UART4, ENABLE); //使能串口;

  }

  void UART4_Send_Byte(u8 Data) //发送一个字节;

  {

  USART_SendData(UART4,Data);

  while( USART_GetFlagStatus(UART4, USART_FLAG_TC) == RESET );

  }

  void UART4_Send_String(u8 *Data) //发送字符串;

  {

  while(*Data)

  UART4_Send_Byte(*Data++);

  }

  void UART4_IRQHandler(void) //中断处理函数;

  {

  u8 res;

  if(USART_GetITStatus(UART4, USART_IT_RXNE) == SET) //判断是否发生中断;

  {

  USART_ClearFlag(UART4, USART_IT_RXNE); //清除标志位;

  res=USART_ReceiveData(UART4); //接收数据;

  UART4_Send_Byte(res); //用户自定义;

  }

  }

  void UART5_Configuration(void)

  {

  GPIO_InitTypeDef GPIO_InitStructure;

  USART_InitTypeDef USART_InitStructure;

  NVIC_InitTypeDef NVIC_InitStructure;

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC|RCC_APB2Periph_GPIOD, ENABLE );

  RCC_APB1PeriphClockCmd(RCC_APB1Periph_UART5, ENABLE );

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12; //UART5 TX;

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

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

  GPIO_Init(GPIOC, &GPIO_InitStructure); //端口C;

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; //UART5 RX;

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

  GPIO_Init(GPIOD, &GPIO_InitStructure); //端口D;

  USART_InitStructure.USART_BaudRate = 9600; //波特率;

  USART_InitStructure.USART_WordLength = USART_WordLength_8b; //数据位8位;

  USART_InitStructure.USART_StopBits = USART_StopBits_1; //停止位1位;

  USART_InitStructure.USART_Parity = USART_Parity_No ; //无校验位;

  USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;

[1] [2]
关键字:stm32  串口  注意事项 引用地址:关于stm32串口必须要学的5个串口以及串口应用和注意事项

上一篇:使用 STM32CubeIDE 开发基于 STM32 的 HID 鼠标或键盘
下一篇:STM32一文通(1) -SWD调试 新建工程

推荐阅读最新更新时间:2026-03-08 10:35

STM32 ADC校准详解:步骤与注意事项
以下是STM32 ADC校准的详细技术说明,包含实际操作步骤和注意事项: 一、ADC校准的必要性 误差来源分析: 零点偏移误差(Offset Error):输入0V时输出不为0 增益误差(Gain Error):满量程时的线性偏差 非线性误差(DNL/INL):转换曲线的阶梯偏差 温度漂移(典型值±2℃时±4LSB) 校准目标: 12位ADC的有效精度达到±1LSB 减少芯片个体差异影响 补偿供电电压波动带来的误差 二、STM32内置校准流程 1. 校准寄存器操作流程 // 适用于STM32F1/F4系列的标准校准流程 void ADC_Calibration(ADC_TypeDef* ADCx) { //
[单片机]
STM32单片机使用注意事项
串口输入乱码 1.target 修改成 8.0 2.stm32Fxx.h 将 //#define HSE_VALUE ((uint32_t)25000000) /*! Value of the External oscillator in Hz */ 改成 #define HSE_VALUE ((uint32_t)8000000) /*! Value of the External oscillator in Hz */
[单片机]
STM32F103串口通讯注意事项
(1)通过串口不断发送数据 库函数中的 void USART_SendData(USART_TypeDef* USARTx, uint16_t Data); 函数不具备缓存作用,当我们需要连续不断发送一段数据时,我们需要等待前一次数据发送完成,放置数据丢失错误 void usart2_send_char(u8 _char) { /* 等待上一次发送完成*/ while( !(USART2- SR & (1 USART2_SR_TC) )) ; USART2- DR = _char ; } (2)通过中断接受数据 通过中断中断接受数据时我们可以通过读取数据寄存器中的值使硬件自动清除挂起的接收中断标志
[单片机]
Keil5配置与新建STM32工程的步骤及注意事项
新建工程文件夹 建立名字为LED的文件夹存放工程文件,其中再建立两个文件夹,Listing文件夹用于存放编译器编译时候产生的C语言、汇编、链接文件,Output文件夹用于存放编译产生的调试信息、hex文件、预览信息、封装库等。 文件夹建好之后,在LED文件夹下存放startup_stm32f10x_hd.s、stm32f10x.h、main.c文件。前两个文件可以在STM32F1xx的固件库中找到,第三个文件是空文件。 在Keil5中新建工程的步骤 如下图所示,Project-New,新建工程,工程名为Led,点击保存。 点击保存之后,弹出如下窗口,选择芯片型号。根据你开发板使用的芯片具体的型号来选择。如果这里没有出
[单片机]
Keil5配置与新建<font color='red'>STM32</font>工程的步骤及<font color='red'>注意事项</font>
AVR串口调试注意事项
这两天一直在调试AVR单片机的UASRT部分。可是不论我如何调试就是无法实现,AVR和PC机的COM1口进行串口通信,我也测试了电压,但是电压一切都很正常。这着实让我头痛了这么几天。 今天我偶然打开AVR单片机的说明书,才让我恍然大悟。呵呵! AVR单片机系统时钟,外围时钟熔丝设定 。 所以想写一点儿关于AVR串口调试事项,结合我这段时间AVR单片机调试经验:   1、就是关于AVR单片机USART通信时钟的设定,这是AVR和其他单片机的重要区别:它的时钟可以通过内带的熔丝为设定,你可以根据需要自行设定,增加设计的灵活性,但对于初学者这也是个让人很容易的地方。因为我们大多都是以MS-51单片机蓝本学
[单片机]
STM32单片机通信协议操作步骤及注意事项
STM32单片机通信协议是一种用于在不同设备之间进行数据传输的协议,它可以帮助设备之间进行高效的通信。STM32单片机通信协议可以用于实现多种不同的应用,如智能家居、智能安防、智能交通等。 STM32单片机通信协议的使用方法主要包括以下几个步骤: 1. 首先,需要确定使用的协议类型,如UART、I2C、SPI等,并确定使用的协议的具体参数,如波特率、数据位、停止位等。 2. 然后,需要在STM32单片机上配置相应的通信接口,并将其与外部设备连接起来。 3. 接着,需要编写相应的程序,实现STM32单片机与外部设备之间的数据传输。 4. 最后,需要将程序烧录到STM32单片机上,并运行程序,实现STM32单片机与外部设备之间的数
[单片机]
stm32官方SD卡驱动的使用注意事项
使用stm32 CUBE 库所提供的SD卡驱动的朋友们请注意: 1、适用于MMC卡,本人没测试过,mmc卡现在已经是收藏品了,谁还会用呢,so,st的团队求求你们严谨的好不好。 2、SD卡初始化过不去,因为丫在发送CMD0后,接着发送了CMD1(CMD1是mmc卡的独有指令,SD卡是不需要发CMD1的),读写驱动函数也有问题(不分情况的修改了sd卡的块大小,统统设置成了512字节),就因为这个害的本人浪费了3 天的时间,无奈只好自己重写了一遍驱动。so,如果要在项目中使用SD卡,须修改驱动程序。 3、其实好多官方例程都有bug,不要太迷信官方的东西。 总之,官方也有坑,使用须谨慎!
[单片机]
STM32实例之LED灯闪烁控制以及相关注意事项
在本实例中,主要是为了实现LED灯的闪烁。首先分析LED的驱动方式,本实验中使用的是OpenM3V,内置8个LED均采用灌流方式驱动(低电平亮)。如果想要实现其闪烁,则需要给相应端口持续不断的高低交替电平。 在软件结构设计中,加入使用LED8,则需要在PD7口不断的输出高电平和低电平。首先需要初始化系统时钟,然后再开始配置PD7作为输出使用,打开外设时钟最后控制PD7输出持续的高低轮流。 开始 - 配置系统时钟 - 配置PD7作为输出在打开PD外设时钟 - 置位PD7,熄灭LED8 - 延时程序 - 清PD7,点亮LED8 - 延时 - 置位PD7,以此开始循环闪烁。 以下给出具体的代码(代码运行在KEIL5上)。 在软件代码编写
[单片机]
小广播
最新单片机文章
何立民专栏 单片机及嵌入式宝典

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

厂商技术中心

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

 
机器人开发圈

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