【STM32】STM32F4中USART的使用方法和Printf的重定义(基于CubeMX和Keil)

本文详细介绍了如何在STM32F407ZGT6单片机上通过STM32CubeMX生成代码,配置KeilIDE以使用MicroLIB重定义printf函数,以及解决串口通信中遇到的问题,包括串口号匹配、MicroLIB启用和调试助手选择等。
摘要由CSDN通过智能技术生成

一、前言

主要目的:利用串口来收发一些数据
使用芯片:STM32F407ZGT6
使用函数:HAL库
使用工具:STM32CubeMX + Keil uVision5
串口工具:XCOM V2.6

二、STM32CubeMX生成代码

2.1 选择芯片

在这里插入图片描述

2.2 配置相关模式

主要配置内容:

  • 调试模式为Serial
  • 设置USART1为异步模式
    在这里插入图片描述
    在这里插入图片描述

2.3 生成代码

自己给工程命名然后选择打开的编译器

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

三、Keil重定义Printf

3.1 勾选“UseMicroLIB”

在魔术棒中打勾“UseMicroLIB”,否则即使我们include了stdio.h且编译器没没有报错的情况下仍然用不了printf。
在这里插入图片描述

3.2 添加头文件和修改fputc和fgetc

usart.c函数里面添加头文件,如下:

/* USER CODE BEGIN 0 */
#include <stdio.h>
/* USER CODE END 0 */

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

usart.c里面添加下面的代码。

/* USER CODE BEGIN 1 */
/**
  * 函数功能: 重定向c库函数printf到DEBUG_USARTx
  * 输入参数: 无
  * 返 回 值: 无
  * 说    明:无
  */
int fputc(int ch, FILE *f)
{
  HAL_UART_Transmit(&huart1, (uint8_t *)&ch, 1, 0xffff);
  return ch;
}
 
/**
  * 函数功能: 重定向c库函数getchar,scanf到DEBUG_USARTx
  * 输入参数: 无
  * 返 回 值: 无
  * 说    明:无
  */
int fgetc(FILE *f)
{
  uint8_t ch = 0;
  HAL_UART_Receive(&huart1, &ch, 1, 0xffff);
  return ch;
}
/* USER CODE END 1 */

在这里插入图片描述

四、测试Printf的效果

4.1 字符串测试

main.c函数的while1添加下面的代码进行测试:

    printf("终于可以使用printf函数了!!!\n");
    HAL_Delay(1000);

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

4.2 格式化输出测试

main.c函数的while1添加下面的代码进行测试:

  /* USER CODE BEGIN 2 */
  float f_num[] = {0.1,0.2,0.3};
  uint8_t i_num[3] = {1,2,3};
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    printf("终于可以使用printf函数了!!!\n");
    HAL_Delay(500);
    printf("试一下格式化输出 ==> 浮点数:%.3f,\t整数:%d\n", f_num[0], i_num[0]);
    HAL_Delay(500);
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */

在这里插入图片描述

五、存在问题的解决方法

5.1 检查串口号是否一致

2.2 配置相关模式中使用的是USART1,因此3.2 添加头文件和修改fputc和fgetc中的这两个函数也是用USART1,如果有问题,看看是否有对应上。
如果你使用的是USART2,那你就把这两个函数里面对应的修改一下,举例一下,就是将下面的huart1改为huart2.

int fputc(int ch, FILE *f)
{
  HAL_UART_Transmit(&huart2, (uint8_t *)&ch, 1, 0xffff);//修改这里的数字
  return ch;
}

5.2 打开MicroLIB

就是3.1 勾选“UseMicroLIB”的这一步,这个也是我解决一直没法使用printf的方法。

5.3 换一个串口调试助手

有可能是串口调试助手显示的问题,换一个试试看。

5.4 复位一下

有时候会忘记复位,程序可能还没启动。

5.5 对比博主的代码

上面的解决方法还是不行,你就对比我的上传的代码吧。
https://download.csdn.net/download/weixin_52296952/88804366

  • 28
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,以下是使用KEIL编写STM32 F4串口输出的示例程序: ```c #include "stm32f4xx.h" #include <stdio.h> // 定义波特率 #define BAUDRATE 115200 // 定义串口号和串口时钟 #define USARTx USART1 #define USARTx_CLK RCC_APB2Periph_USART1 // 定义串口GPIO引脚和时钟 #define USARTx_TX_GPIO_PORT GPIOA #define USARTx_TX_GPIO_CLK RCC_AHB1Periph_GPIOA #define USARTx_TX_PIN GPIO_Pin_9 #define USARTx_TX_SOURCE GPIO_PinSource9 #define USARTx_RX_GPIO_PORT GPIOA #define USARTx_RX_GPIO_CLK RCC_AHB1Periph_GPIOA #define USARTx_RX_PIN GPIO_Pin_10 #define USARTx_RX_SOURCE GPIO_PinSource10 // 初始化串口函数 void USART_Config(void) { USART_InitTypeDef USART_InitStructure; GPIO_InitTypeDef GPIO_InitStructure; // 使能串口和GPIO时钟 RCC_APB2PeriphClockCmd(USARTx_CLK, ENABLE); RCC_AHB1PeriphClockCmd(USARTx_TX_GPIO_CLK | USARTx_RX_GPIO_CLK, ENABLE); // 将GPIO端口配置为复用功能 GPIO_PinAFConfig(USARTx_TX_GPIO_PORT, USARTx_TX_SOURCE, GPIO_AF_USART1); GPIO_PinAFConfig(USARTx_RX_GPIO_PORT, USARTx_RX_SOURCE, GPIO_AF_USART1); // 配置GPIO端口为输出模式 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; // 配置TX引脚 GPIO_InitStructure.GPIO_Pin = USARTx_TX_PIN; GPIO_Init(USARTx_TX_GPIO_PORT, &GPIO_InitStructure); // 配置RX引脚 GPIO_InitStructure.GPIO_Pin = USARTx_RX_PIN; GPIO_Init(USARTx_RX_GPIO_PORT, &GPIO_InitStructure); // 配置串口参数 USART_InitStructure.USART_BaudRate = BAUDRATE; USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx; // 初始化串口 USART_Init(USARTx, &USART_InitStructure); // 使能串口 USART_Cmd(USARTx, ENABLE); } // 定向printf函数到串口输出 int fputc(int ch, FILE *f) { USART_SendData(USARTx, (uint8_t)ch); while (USART_GetFlagStatus(USARTx, USART_FLAG_TXE) == RESET); return ch; } int main(void) { // 初始化串口 USART_Config(); // 输出字符串 printf("Hello, world!\n"); // 进入死循环 while (1); } ``` 在上述代码,我们首先定义串口号、GPIO引脚、波特率等参数,然后在`USART_Config`函数配置了串口和GPIO的相关参数,最后在`main`函数输出了一句字符串并进入死循环。 需要注意的是,为了能够使用printf函数输出字符串,我们还需要在`fputc`函数定向输出到串口
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值