1.printf重定向简介
在C语言中printf函数里,默认输出设备是显示器,如果想要用这个函数将输出结果到串口或者LCD上显示,就必须重定义标准库函数里中printf函数调用的与输出设备相关的函数。
比如要使用printf输出到串口,需要先将fputc函数里面的输出指向串口,这个更改就叫重定向。
那么如何让STM32使用printf函数呢?
int fputc(int ch,FILE *p) //在使用printf函数时会自动调用这个函数
{
USART_SendData(USART1,(u8)ch); //将此处原来的显示器改为USART1
while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);
return ch;
}
2.printf函数格式
printf("<格式化字符串>", <参量表>);
常用格式化规定字符如下:
%d 按照十进制整型数打印
%6d 按照十进制整型数打印,至少6个字符宽
%f 按照浮点数打印
%6f 按照浮点数打印,至少6个字符宽
%.2f 按照浮点数打印,小数点后有2位小数
%6.2f 按照浮点数打印,至少6个字符宽,小数点后有2位小数
%x 按照十六进制打印
%c 打印字符
%s 打印字符串
例如:使用printf函数输出一个整型数据1234,则调用格式如下:
int data=1234;
printf(“输出整型数据data=%d\r\n”,data);
main.c
#include "system.h"
#include "led.h"
#include "SysTick.h"
#include "usart.h"int main()
{u8 i;u16 data=1234;float fdata=12.3456;char str[]="Hello World!";SysTick_Init(72);NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//中断优先级分组LED_Init();USART1_Init(9600);while(1){i++;if(i%20 ==0){led1=!led1;//LED1闪,用来指示主程序循环是否运行printf("输出整型数data=%d\r\n",data);printf("输出浮点数fdata=%.2f\r\n",fdata);printf("输出十六进数data=%X\r\n",data);printf("输出八进数data=%o\r\n",data);printf("输出子符串str=%s\r\n",str);}delay_ms(10);}
}
#include "usart.h"int fputc(int ch,FILE *p) //在使用printf函数时自动调用此函数
{USART_SendData(USART1,(u8)ch); while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);return ch;
}void USART1_Init(u32 BoudRate)
{GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure;NVIC_InitTypeDef NVIC_InitStructure;RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//使能USART1时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE); //使能GPIOA时钟GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9; //PA9为USART1的TXDGPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;//GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; //GPIO_Init(GPIOA,&GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10; //PA10为USART1的RXDGPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING;//GPIO_Init(GPIOA,&GPIO_InitStructure);USART_InitStructure.USART_BaudRate = BoudRate;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(USART1, &USART_InitStructure);USART_Cmd(USART1, ENABLE); //使能串口1USART_ClearFlag(USART1, USART_FLAG_TC);//清除串口1发送中断标志位USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启RXNE(接收中断)//设置中断优先级,使能中断通道NVIC_InitStructure.NVIC_IRQChannel= USART1_IRQn;NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority =3;NVIC_InitStructure.NVIC_IRQChannelSubPriority =3;NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;NVIC_Init(&NVIC_InitStructure);}void USART1_IRQHandler(void)
{u8 r;if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET){r=USART_ReceiveData(USART1);//开发板接收从电脑发过来的数据USART_SendData(USART1, r);//开发板再把收到的数据发送回电脑while(USART_GetFlagStatus(USART1, USART_FLAG_TC)!=SET);//等待,直到发送完成USART_ClearFlag(USART1, USART_FLAG_TC);//清除串口1发送中断标志位}}
实验是成功的,在开发板上运行后结果如下: