STM32学习和实践笔记(28):printf重定向实验

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发送中断标志位
		
	}
	

	
}

实验是成功的,在开发板上运行后结果如下:

                    

  • 8
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

复业思维20240108

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

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

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

打赏作者

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

抵扣说明:

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

余额充值