第一种方法:使用标准C库,但使用标准C库你必须关闭半主机模式

(1)添加下面代码就是关闭半主机模式

/* 告知连接器不从C库链接使用半主机的函数 */
#pragma import(__use_no_semihosting)/* 定义 _sys_exit() 以避免使用半主机模式 */
void _sys_exit(int x)
{x = x;
}/* 标准库需要的支持类型 */
struct __FILE
{int handle;
};FILE __stdout;

在独立应用程序中,不可能支持半主机操作。 因此,必须确保应用程序中没有链接 C 库半主机函数。

为确保没有从 C 库链接使用半主机的函数, 必须导入符号 __use_no_semihosting 。

可在工程的任何 C 或汇编语言源文件中执行此操作,如下所示:

在 C 模块中,使用 #pragma 指令:

#pragma import(__use_no_semihosting)

在汇编语言模块中,使用 IMPORT 指令:

IMPORT __use_no_semihosting

(2)串口重定向
将你要输出信息的串口添加到这句函数里面
///< 串口发送重定向
int fputc(int ch, FILE * file)
{Uart_SendDataPoll(M0P_UART0,ch);         //调用库函数,通过UART0发送一个字母。return ch;
}
如果是不同型号的MCU,或者使用那个串口更改对应的串口号即可
接下来你就自己配置好对应的串口初始化就OK了

这里需要注意下:本文代码我采用华大HC32L系列的,它这个库函数Uart_SendDataPoll(M0P_UART0,ch)里面是有等待数据发送完毕的

如果你用的是STM32单片机,重定向函数应该这么写

//< 串口发送重定向
int fputc(int ch, FILE * file)
{USART_SendData(USART1,ch);       while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束return ch;
}

需要自己在添加一行代码来等待发送完毕,因为ST发送库函数里面没有等待语句

(3)下面是我自己完整的配置(我是使用串口0)
#include "BSP_GPIO.h"
#include "BSP_Uart.h"//#include "UFD.h"
uint8_t u8Rx0Data;/* 告知连接器不从C库链接使用半主机的函数 */
#pragma import(__use_no_semihosting)/* 定义 _sys_exit() 以避免使用半主机模式 */
void _sys_exit(int x)
{x = x;
}/* 标准库需要的支持类型 */
struct __FILE
{int handle;
};FILE __stdout;
///< 串口发送重定向
int fputc(int ch, FILE * file)
{Uart_SendDataPoll(M0P_UART0,ch);         //调用库函数,通过UART0发送一个字母。return ch;
}
static void Uart0_PortInit(void)
{stc_gpio_cfg_t stcGpioCfg;///< 打开GPIO外设时钟门控Sysctrl_SetPeripheralGate(SysctrlPeripheralGpio, TRUE);DDL_ZERO_STRUCT(stcGpioCfg);///< 端口方向配置 UART0_TXDstcGpioCfg.enDir = GpioDirOut;Gpio_Init(PORT_DEBUG_TXD,PIN_DEBUG_TXD,&stcGpioCfg);Gpio_SetAfMode(PORT_DEBUG_TXD,PIN_DEBUG_TXD,GpioAf2);//UART0_TXD///< 端口方向配置 UART0_RXDstcGpioCfg.enDir = GpioDirIn;///< 端口上下拉配置->上拉stcGpioCfg.enPu = GpioPuEnable;Gpio_Init(PORT_DEBUG_RXD,PIN_DEBUG_RXD,&stcGpioCfg);Gpio_SetAfMode(PORT_DEBUG_RXD,PIN_DEBUG_RXD,GpioAf2);//UART0_RXDSysctrl_SetFunc(SysctrlSWDUseIOEn, TRUE);    /*Set SWD port to GPIO mode*/
}
static void Uart0_Init(void)
{stc_uart_cfg_t  stcCfg;stc_uart_baud_t stcBaud;DDL_ZERO_STRUCT(stcCfg);DDL_ZERO_STRUCT(stcBaud);///< 打开UART0外设时钟门控Sysctrl_SetPeripheralGate(SysctrlPeripheralUart0,TRUE);///<UART InitstcCfg.enRunMode        = UartMskMode1;                 ///<模式1stcCfg.enStopBit        = UartMsk1bit;                  ///<1bit停止位stcCfg.enMmdorCk        = UartMskDataOrAddr;            ///<多机模式时stcCfg.stcBaud.u32Baud  = 1000000;                      ///<波特率1000000stcCfg.stcBaud.enClkDiv = UartMsk8Or16Div;              ///<通道采样分频配置stcCfg.stcBaud.u32Pclk  = Sysctrl_GetPClkFreq();        ///</<获得外设时钟(PCLK)频率值Uart_Init(M0P_UART0, &stcCfg);                          ///<串口初始化///<UART中断使能Uart_ClrStatus(M0P_UART0,UartRC);                       ///<清接收请求Uart_ClrStatus(M0P_UART0,UartTC);                       ///<清接收请求Uart_EnableIrq(M0P_UART0,UartRxIrq);                    ///<使能串口接收中断EnableNvic(UART0_2_IRQn, IrqLevel3, TRUE);              ///<系统中断使能
}void BSP_UartInit(void)
{Uart0_PortInit();Uart0_Init();
}

第二种方法:使用微库,因为使用微库的话 ,不会使用半主机模式,咱也就不用在写那几句关闭半主机模式的语句

(1)在Keil工程中“中勾选 ”Use MicroLIB
(2)重定向输出(这个跟第一种方法一样)
//< 串口发送重定向
int fputc(int ch, FILE * file)
{Uart_SendDataPoll(M0P_UART0,ch);         //调用库函数,通过UART0发送一个字母。return ch;
}
(3)下面是我自己完整的配置(我是使用串口0)
#include "BSP_GPIO.h"
#include "BSP_Uart.h"//#include "UFD.h"
uint8_t u8Rx0Data;//< 串口发送重定向
int fputc(int ch, FILE * file)
{Uart_SendDataPoll(M0P_UART0,ch);         //调用库函数,通过UART0发送一个字母。return ch;
}
static void Uart0_PortInit(void)
{stc_gpio_cfg_t stcGpioCfg;///< 打开GPIO外设时钟门控Sysctrl_SetPeripheralGate(SysctrlPeripheralGpio, TRUE);DDL_ZERO_STRUCT(stcGpioCfg);///< 端口方向配置 UART0_TXDstcGpioCfg.enDir = GpioDirOut;Gpio_Init(PORT_DEBUG_TXD,PIN_DEBUG_TXD,&stcGpioCfg);Gpio_SetAfMode(PORT_DEBUG_TXD,PIN_DEBUG_TXD,GpioAf2);//UART0_TXD///< 端口方向配置 UART0_RXDstcGpioCfg.enDir = GpioDirIn;///< 端口上下拉配置->上拉stcGpioCfg.enPu = GpioPuEnable;Gpio_Init(PORT_DEBUG_RXD,PIN_DEBUG_RXD,&stcGpioCfg);Gpio_SetAfMode(PORT_DEBUG_RXD,PIN_DEBUG_RXD,GpioAf2);//UART0_RXDSysctrl_SetFunc(SysctrlSWDUseIOEn, TRUE);    /*Set SWD port to GPIO mode*/
}
static void Uart0_Init(void)
{stc_uart_cfg_t  stcCfg;stc_uart_baud_t stcBaud;DDL_ZERO_STRUCT(stcCfg);DDL_ZERO_STRUCT(stcBaud);///< 打开UART0外设时钟门控Sysctrl_SetPeripheralGate(SysctrlPeripheralUart0,TRUE);///<UART InitstcCfg.enRunMode        = UartMskMode1;                 ///<模式1stcCfg.enStopBit        = UartMsk1bit;                  ///<1bit停止位stcCfg.enMmdorCk        = UartMskDataOrAddr;            ///<多机模式时stcCfg.stcBaud.u32Baud  = 1000000;                      ///<波特率1000000stcCfg.stcBaud.enClkDiv = UartMsk8Or16Div;              ///<通道采样分频配置stcCfg.stcBaud.u32Pclk  = Sysctrl_GetPClkFreq();        ///</<获得外设时钟(PCLK)频率值Uart_Init(M0P_UART0, &stcCfg);                          ///<串口初始化///<UART中断使能Uart_ClrStatus(M0P_UART0,UartRC);                       ///<清接收请求Uart_ClrStatus(M0P_UART0,UartTC);                       ///<清接收请求Uart_EnableIrq(M0P_UART0,UartRxIrq);                    ///<使能串口接收中断EnableNvic(UART0_2_IRQn, IrqLevel3, TRUE);              ///<系统中断使能
}void BSP_UartInit(void)
{Uart0_PortInit();Uart0_Init();
}

总结:

要使用第一种还是第二种,看自己选择,大家好像选择第一种比较多。

华大单片机、STM32单片机如何做printf串口打印格式化输出相关推荐

  1. 华大单片机HC32L136如何做printf串口打印格式化输出

    目录 写在前面 半主机模式介绍(可以跳过) 串口接收初始化 串口发送重定向 主程序 连好开发板 运行效果 写在前面 通常工程师在做产品开发和程序调试的时候会通过仿真器进入仿真模式来获得芯片运行的结果, ...

  2. 51单片机 DHT11温湿度传感器LCD显示+串口打印显示

    第一部分:DHT11详解: DHT11是一款有已校准数字信号输出的温湿度传感器. 精度湿度±5%RH, 温度±2℃,量程湿度20-90%RH, 温度0~50℃. **一.电路连接分析 1.引脚图** ...

  3. shell printf命令:格式化输出语句

    printf 命令用于格式化输出, 是echo命令的增强版.它是C语言printf()库函数的一个有限的变形,并且在语法上有些不同. 注意:printf 由 POSIX 标准所定义,移植性要比 ech ...

  4. STM32移植时printf串口打印中文乱码,mark一下

    STM32移植后,使用printf进行串口打印,中文显示乱码: 移植时是先新建.txt文档,复制完代码后改为.c文件,查找相关资料,得知.c文件为"UTF-8"编码,需要将编码格式 ...

  5. C语言printf()格式化输出

    printf()函数 请求printf()函数打印数据的指令要与待打印数据的类型相匹配.例如,打印整数时使用%d,打印字符时使用%c.这些符号被称为转换说明(conversion specificat ...

  6. 使用单片机的串口打印功能

    转载请注明出处 个人看法,仅供分享 单片机调试手段: 1.在线调试 部分芯片支持该功能,典型的有STM32系列的SWD接口,沁恒的CH32V和CH32F系列.CH56X部分RISC-V的两线调试. 2 ...

  7. STM32单片机实现DMA+ADC+UART功能

    突然想测试一下STM32单片机ADC采样速率问题,按照常规方法,可以通过ADC采样,然后将采样值打印出来.但是这种方法在处理和打印数据的时候会占用很多时间,导致处理数据的时间超过了ADC的采样时间.于 ...

  8. 1.国民技术N32G45X例程之-串口打印

    国民技术N32G45X例程之-串口打印 提示:use MicroLIB,printf串口打印 文章目录 前言 一.国民技术N32G45X串口配置 二.printf函数 1.国民技术N32G45X官方库 ...

  9. matlab printf格式化输出,Shell printf格式化输出命令

    printf 是 awk 的重要格式化输出命令,本节我们先介绍一下 printf 命令如何使用. 需要注意,在 awk 中可以识别 print 输出动作和 printf 输出动作(区别是:print ...

最新文章

  1. 线程安全操作HashMap
  2. python网络编程:UDP方式传输数据
  3. VC++中,如何定义callback函数和它的触发事件?
  4. 在线课程学习、科研科技视频网站
  5. 面向智能电网的电力大数据存储与分析应用
  6. Spring分页实现PageImplT类
  7. openwrt patch文件怎么用_openwrt 打补丁方式修改内核源码
  8. windows优化大师怎么用_软件不能用又卸载不掉怎么办——用Windows自家的卸载工具吧(dos级卸载)...
  9. 申请企业邮箱有什么流程?
  10. Hadoop job任务分配
  11. jackson 忽略多余字段_Java进阶学习:JSON解析利器JackSon
  12. 个人主页网页设计模板
  13. Windows系统压缩卷时可压缩空间远小于实际剩余空间解决方法
  14. JanusGraph入坑笔记(五)- GraphTraversal (Vertex centric)
  15. 解放生产力 - Xcode 与 Simulator 技巧整理(持续更新中)
  16. [Android系统开发]Launcher Hotseat图标居中排列
  17. 工作感受月记 201909月
  18. DXC Technology任命Ken Corless为产品与战略合作伙伴执行副总裁
  19. 华为云数据库 GaussDB(for MySQL),让企业无忧数据恢复
  20. 欧姆龙c200he基本语言,欧姆龙C200HE-CPU11-E操作手册 - 广州凌控

热门文章

  1. C语言把mac地址转为6字节数组,mac地址十六进制转十进制
  2. 第十一届北京师范大学程序设计竞赛(网络同步赛)+A. BNU ACM校队时间安排表
  3. Java项目:企业绩效考核管理系统(java+SSM+JSP+bootstrap+jQuery+Mysql)
  4. Oracle项目管理系统之概算模板
  5. emqx开启自定义权限认证
  6. 单一世界架构初探之边界冲突
  7. 如何用软件测试交易系统的胜率,交易评测系统简单设置,优质指标带你百战百胜,全场胜率一键计算...
  8. 电子邮件归档普及迅速 面临应用瓶颈
  9. Ubuntu下缺少无线网卡驱动或者无限网卡驱动无法使用的解决方案
  10. C++输入密码回显星号