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

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

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、前言

主要目的:利用串口来收发一些数据
使用芯片: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

### Keil 5 中重定义错误解决方案 在编译过程中遇到 `redefinition` 错误通常意味着某个实体(变量、函数或宏)被多次声明或定义。这类问题可能由多种原因引起,包括但不限于头文件重复包含、命名冲突以及库文件之间的不兼容。 #### 头文件保护机制 为了避免因多个源文件包含相同的头文件而导致的重定义错误,在每个 `.h` 文件中应使用预处理指令来防止多重包含: ```c #ifndef _HEADER_FILE_NAME_H_ #define _HEADER_FILE_NAME_H_ // Header file content goes here #endif /* _HEADER_FILE_NAME_H_ */ ``` 这种做法可以有效阻止同一头文件在一个翻译单元内被多次加载[^1]。 #### 检查全局变量函数声明 确保所有的外部链接符号只在一个地方声明为定义形式 (`extern`) 或者初始化。如果两个不同的源文件都尝试定义同一个具有相同名称的空间,则会发生冲突。对于静态库项目而言,还需确认不同模块间是否存在同名但意义各异的对象。 #### 链接器设置审查 有时即使解决了上述问题仍然会碰到此类报错,这可能是由于链接阶段出现了重复的目标文件或者是第三方库之间存在版本差异所造成的。此时应该仔细核对项目的 **Linker Settings**, 特别关注是否有不必要的对象文件被列入到了最终可执行程序构建路径之中。 #### 清理并重建工程 当修改了配置选项之后,建议先清理整个工作区再重新编译全部代码,这样能消除残留旧目标文件所带来的潜在影响。 ```bash make clean make all ``` 通过以上措施往往能够有效地定位并修复大多数常见的 C/C++ 编程环境下的 “redefinition” 类型编译期异常情况。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值