STM32使用printf发送字符串到串口 (重定义printf函数)

问题:在使用STM32调试时,经常使用串口发送信息,为了方便调试与串口发送信息,用printf()函数实现通过串口打印信息。

方法一:

1.添加包含printf()函数的头文件:#include “stdio.h” 
2.重写 stdio.h 头文件中的 int fputc(int ch, FILE *f) 函数

int fputc(int ch, FILE *f)
{
    //等待先前数据传输到移位寄存器
    while( !(USART1->SR & (1 << 7)) );
    //发送字符
    USART_SendData(USART1, (uint8_t) ch);
    return ch;
}
//示例函数中使用了USART1来发送消息
//实际情况可以根据硬件板来决定使用哪个串口
注意:使用while循环先等待先前的字符数据传输到移位寄存器,避免造成字符串首字符发送丢失的问题。 
3.将该函数” int fputc(int ch, FILE *f) “放在main()函数能够调用到的文件中,KEIL->Options for Target’xxx’->Target->Code Generation,勾选Use MicroLIB

方法二:

//加入以下代码,支持printf函数,而不需要选择use MicroLIB    
#if 1
#pragma import(__use_no_semihosting)             
//标准库需要的支持函数                 
struct __FILE 
{
    int handle; 
 
};
 
FILE __stdout;       
//定义_sys_exit()以避免使用半主机模式    
_sys_exit(int x) 
{
    x = x; 
}
//重定义fputc函数 
int fputc(int ch, FILE *f)
{      
    while( !(USART1->SR & (1 << 7)) );
    USART_SendData(USART1,(uint8_t)ch); 
    return ch;
}
#endif 
 

 


方法三(不建议使用)

//加入以下代码,并且勾选使用MiroLIB

int fputc(int ch,FILE *f)
{
    //将Printf内容送往串口
    USART_SendData(USART1,(unsigned char)ch);
    //等待发送完成
    while(USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
    return (ch);
}

使用以上任一方法,均可实现printf()函数打印字符串重定义到USART1。

如果不想从USART1输出数据,可以更改USART1为USART2、3,都可以进行数据打印输出。

  • 5
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
使用printf函数可以将接收到的字符串打印到串口或者调试助手上,以下是一个示例代码: ``` #include "stdio.h" #include "string.h" #include "stm32f4xx.h" #define RX_BUFFER_SIZE 512 uint8_t rx_buffer[RX_BUFFER_SIZE]; // 定义接收缓存区 uint16_t rx_counter = 0; // 定义接收计数器 void USART2_IRQHandler(void) { if (USART_GetITStatus(USART2, USART_IT_RXNE) != RESET) { uint8_t data = USART_ReceiveData(USART2); rx_buffer[rx_counter++] = data; if (data == '\n' || rx_counter >= RX_BUFFER_SIZE) { rx_buffer[rx_counter] = '\0'; // 在字符串末尾添加'\0' printf("Received string: %s\n", rx_buffer); rx_counter = 0; // 接收计数器清零 } } } int main(void) { // 初始化串口 USART_InitTypeDef USART_InitStructure; GPIO_InitTypeDef GPIO_InitStructure; RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE); GPIO_PinAFConfig(GPIOA, GPIO_PinSource2, GPIO_AF_USART2); GPIO_PinAFConfig(GPIOA, GPIO_PinSource3, GPIO_AF_USART2); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3; 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; GPIO_Init(GPIOA, &GPIO_InitStructure); USART_InitStructure.USART_BaudRate = 115200; 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_Rx | USART_Mode_Tx; USART_Init(USART2, &USART_InitStructure); USART_Cmd(USART2, ENABLE); while (1) { // 主循环进行其他操作 } } ``` 在串口断处理函数,当接收到字符'\n'时或者接收计数器达到缓存区大小时,将接收缓存区字符串打印到串口或者调试助手上,然后将接收计数器清零。 需要注意的是,在使用printf函数进行调试时,需要在工程设置开启printf定向功能。具体操作方法可以参考相关资料或者开发板的用户手册。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

蜻蜓队长c

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值