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的TXD
GPIO_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的RXD
GPIO_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); //使能串口1
USART_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发送中断标志位
}
}
实验是成功的,在开发板上运行后结果如下: