stm32串口——printf重定向两种方法

介绍通过printf重定向发送字符串,在串口调试助手上显示字符串

一、代码方法(不用微库)

添加程序到文件中

代码

#pragma import(__use_no_semihosting)             
//标准库需要的支持函数                 
struct __FILE 
{ 
	int handle; 
}; 

FILE __stdout;       
//定义_sys_exit()以避免使用半主机模式    
void _sys_exit(int x) 
{ 
	x = x; 
}
//重定义fputc函数 
int fputc(int ch, FILE *f)
{      
	while((USART1->SR&0X40)==0);//循环发送,直到发送完毕   
    USART1->DR = (u8) ch;      
	return ch;
}

此处使用的是usart1,需要串口初始化

void uart_init(u32 bound){
	//GPIO端口设置
	GPIO_InitTypeDef GPIO_InitStructure;
	USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	 
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);	//使能USART1,GPIOA时钟
  
	//USART1_TX   GPIOA.9初始化
  	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
  	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;	//复用推挽输出
  	GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.9
   
  	//USART1_RX	  GPIOA.10初始化
  	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;//PA10
  	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
  	GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.10  

 	//Usart1 NVIC 配置
  	NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;		//子优先级3
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;			//IRQ通道使能
	NVIC_Init(&NVIC_InitStructure);	//根据指定的参数初始化VIC寄存器
  
   	//USART 初始化设置

	USART_InitStructure.USART_BaudRate = bound;//串口波特率
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
	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); //初始化串口1
  	USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启串口接受中断
  	USART_Cmd(USART1, ENABLE);                    //使能串口1 
}

实物部分:
将ch340与c8t6的usart1连接
将c8t6的boot0置1,boot1置0
将ch340连接电脑
打开flymcu点击开始编程
将c8t6的复位键按一下
程序下载成功

c8t6串口1引脚图:

PA9连ch340的RX脚
PA10连ch340的TX脚
c8t6串口1引脚

二、微库法

在keil中的option中勾选Use MicroLIB

在这里插入图片描述

/**
***********************************************************
* @brief printf函数默认打印输出到显示器,如果要输出到串口,
		 必须重新实现fputc函数,将输出指向串口,称为重定向
* @param
* @return 
***********************************************************
*/
int fputc(int ch, FILE *f)
{
	usart_data_transmit(g_uartHwInfo.uartNo, (uint8_t)ch);//串口发送初始化
	while (RESET == usart_flag_get(g_uartHwInfo.uartNo, USART_FLAG_TBE));//发送结束标志
	return ch;
}
STM32微控制器通常使用标准C库中的printf函数进行输出,但在一些情况下,可能需要重定义或自定义printf函数的行为,特别是在嵌入式开发中,资源有限,或者需要特殊格式化输出时。在STM32中,你可能会遇到这样的需求,例如: 1. 串口通信:printf输出通常通过UART、USART或其他串行接口发送到外部设备,可能需要修改为直接通过硬件发送数据。 2. 缓存控制:为了节省内存,可以重定义printf函数以缓存输出,仅在需要的时候再发送。 3. 扩展输出格式:自定义printf格式字符串以适应特定的应用需求。 在STM32中,你通常可以通过以下几个步骤来重定义printf函数: - 包含头文件`stdio.h`,但不包含`stdarg.h`,因为该头文件包含了printf的声明,你需要自己实现。 - 定义一个函数,比如`my_printf`,原型应为`void my_printf(const char *format, ...)`,这遵循了printf函数的标准参数列表。 - 实现`my_printf`函数,根据`format`参数解析并处理输入的格式字符串和变量,然后选择适当的机制(如串口发送)将信息输出。 例如,下面是一个简单的示例: ```c #include "stm32f10x.h" // 或者其他STM32库 void my_printf(const char *format, ...) { va_list args; int i; va_start(args, format); // 初始化变量参数指针 while (*format) { // 遍历格式字符串 if (*format == '%') { // 当遇到格式说明符% char specifier = *(format + 1); // 读取下一个字符作为说明符 // 根据specifier处理不同的输出情况 // ... } else { // 字符不是说明符,简单复制到缓冲区 // ... } format++; // 移动到下一个字符 } // 输出缓冲区内容 // ... va_end(args); // 清理变量参数指针 } // 在你的代码中,用my_printf替换printf void main() { my_printf("Hello, world!\n"); } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值