0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

【经验分享】单片机中如何实现printf输出log?

嵌入式物联网开发 来源:嵌入式物联网开发 作者:嵌入式物联网开发 2022-12-08 08:42 次阅读

【经验分享】C51单片机中如何实现printf输出log?

你在真实的项目工程开发中,有考虑过在类似C51单片机中实现printf输出log吗?本文给出一种参考实现。

1 需求说明2 源码实现2.1 函数申明2.2 功能实现3 源码测试4 小小总结5 更多分享

1 需求说明

这个需求比较简单,就是要在C51单片机中实现printf函数,并使用它来打印输出常用的几种类型的数据,比如整型数据,字符串数据等等。

2 源码实现

2.1 函数申明

通过查看man帮助,我们可以知道printf函数的功能及其简要申明。

2.2 功能实现

以下是我的一个简单实现源码,仅供参考:

#include 
 #include "log.h"
 #include "stdarg.h"
 #include "types.h"static  xdata char  Simple_Prn_Buf[6]; 
 uchar Align_Bit = 0;
 ​
 void Dark_Fill_String(void)
 {
     unsigned char i, j;
 ​
     j = 4;
     for(i=0;i<5;i++)
     {
         if (Simple_Prn_Buf[i] != '0')
         {
             j = i;
             break;
         }
     }
 ​
     if (j != 0)
     {
         for (i=j;i<6;i++)
             Simple_Prn_Buf[i-j] = Simple_Prn_Buf[i];
     }
     
     j = strlen(Simple_Prn_Buf);
 ​
     if (Align_Bit>j)
     {
         for (i=0;i1;i++)
             Simple_Prn_Buf[Align_Bit-i] = Simple_Prn_Buf[j-i];
 ​
         for (i=0;i'0';
     }
 }
 ​
 void IntToStr(int Int_Data)
 {
     unsigned char i;
     int Shang, Div_Data;
 ​
     Shang = Int_Data;
     Div_Data = 10000;
 ​
     for(i=0;i<5;i++)
     {
         Simple_Prn_Buf[i] = Shang / Div_Data + '0';
         Shang = Shang % Div_Data;
         Div_Data /= 10;
     }
     Simple_Prn_Buf[5] = '';
     Dark_Fill_String();
 }
 ​
 void HexToStr(int Int_Data, unsigned char x)
 {
     unsigned char i;
 ​
     if (Int_Data == 0) {
         if (Align_Bit == 0) {
             Align_Bit = 1;
         } 
         memset(Simple_Prn_Buf, '0', Align_Bit);
         Simple_Prn_Buf[Align_Bit] = '';
         return;
     }
 ​
     x = (x) ? 'A' : 'a';
 ​
     for (i=0;i<4;i++)
     {
         Simple_Prn_Buf[i] = ((Int_Data >> (3-i)*4)) & 0x000F;
 ​
         if (Simple_Prn_Buf[i] > 9)
             Simple_Prn_Buf[i] += (x - 10);
         else
             Simple_Prn_Buf[i] += '0';
     }
 ​
     Simple_Prn_Buf[4] = '';
     Dark_Fill_String(); 
 }
 ​
 int xprintf(char *fmt, ...)
 {
     char *Str;
     int  Int_Data;
     uchar Fill_Flag = 0;
 ​
     va_list ap;
     va_start(ap, fmt);
 ​
     while(*fmt)
     {
         if ((*fmt != '%') && (Fill_Flag == 0))
         {
             Push_To_TX_Buffer(*fmt++);
             continue;
         }
 ​
         if (*fmt == '%')
         {
             fmt++;
             Align_Bit = 0;
             Fill_Flag = 1;
         }
         
         switch(*fmt)
         {
             case 's':
                  Str = va_arg(ap, char *);
                  for (; *Str; Str++)
                     Push_To_TX_Buffer(*Str);
                  Fill_Flag = 0;
                  Align_Bit = 0;
                  break;
 ​
             case 'd':
                  Int_Data = va_arg(ap, int);
                  IntToStr(Int_Data);
                  for (Str=Simple_Prn_Buf; *Str; Str++) {
                     Push_To_TX_Buffer(*Str);
                  }
                  Fill_Flag = 0;
                  Align_Bit = 0;
                  break;
 ​
             case 'x':
                  Int_Data = va_arg(ap, int);
                  HexToStr(Int_Data, 0); //小写
                  for (Str=Simple_Prn_Buf; *Str; Str++) {
                     Push_To_TX_Buffer(*Str);
                  }
                  Fill_Flag = 0;
                  Align_Bit = 0;
                  break;
 ​
             case 'X':
                  Int_Data = va_arg(ap, int);
                  HexToStr(Int_Data, 1); //大写
                  for (Str=Simple_Prn_Buf; *Str; Str++) {
                     Push_To_TX_Buffer(*Str);
                  }
                  Fill_Flag = 0;
                  Align_Bit = 0;
                  break;
 ​
             default:
                  //Push_To_TX_Buffer(*fmt);
                  Align_Bit = *fmt - '0';
                  if (Align_Bit > 9)
                     Align_Bit = 9;
                  break;
         }
         fmt++;
     }
     va_end(ap);
 ​
     return 0;
 }
 ​

3 源码测试

简单的测试代码如下:

#include 
 #include 
 #include 
 ​
 extern int xprintf(const char* format, ...);
 ​
 #define LOG(fmt, arg...)xprintf(fmt, ##arg)int main(int argc, const char *argv[])
 {
 //uart_init();
     
 #if 1
 puts("Hello World

");/*
 unsigned int size1 = sizeof(char *);
 unsigned int size2 = sizeof(int *);
 unsigned int size3 = sizeof(int);
 unsigned int size4 = sizeof(short int);
 size = sizeof(int);
 printf("sizeof int = %d
", size);
 size = sizeof(short int);
 printf("sizeof short int = %d
", size);*/
 uart2_send_string("Hello World

");
 uart1_send_string("



");
 uart1_send_string("



");
 LOG("1Test log

");
 LOG("2Test log %c

", '=');
 LOG("3Test log %s %s

", "123", "098");
 LOG("4--Test log %d

", -456);
 LOG("4Test log %d

", 456);
 LOG("5Test log %u

", 789);
 LOG("6Test log %x

", 0x12);
 LOG("6Test log %x

", 0x1A);
 LOG("6Test log %x

", 0x1A);
 LOG("6Test log %x

", 0x1B);
 LOG("6Test log %x

", 0xab);
 LOG("6Test log %x

", 0xAB);
 LOG("6Test log %x

", 0x01);
 LOG("6Test log %2x

", 0x01);
 LOG("6Test log %x

", 0x00);
 LOG("6Test log %2x

", 0x00);
 #endifreturn 0;
 }

感兴趣的朋友可以把这段测试代码,在C51平台上编译运行下,相信它会给你惊喜的!

4 小小总结

printf函数看似很常用,但是真正到了要自己去实现它的时候,你又会发现其实还是蛮多东西需要考虑的。

同时,即便是本文中的实现,还是有些类型的数据是不支持输出的,比如 long int 类型这种,就比较难输出;还有 float类型这种数据,也是没法输出的。

看到这里,你是否还有更好的实现方案呢?

5 更多分享

[架构师李肯]

架构师李肯全网同名 ),一个专注于嵌入式IoT领域的架构师。有着近10年的嵌入式一线开发经验,深耕IoT领域多年,熟知IoT领域的业务发展,深度掌握IoT领域的相关技术栈,包括但不限于主流RTOS内核的实现及其移植、硬件驱动移植开发、网络通讯协议开发、编译构建原理及其实现、底层汇编及编译原理、编译优化及代码重构、主流IoT云平台的对接、嵌入式IoT系统的架构设计等等。拥有多项IoT领域的发明专利,热衷于技术分享,有多年撰写技术博客的经验积累,连续多月获得RT-Thread官方技术社区原创技术博文优秀奖,荣获[CSDN博客专家]、[CSDN物联网领域优质创作者]、[2021年度CSDN&RT-Thread技术社区之星]、[2022年RT-Thread全球技术大会讲师]、[RT-Thread官方嵌入式开源社区认证专家]、[RT-Thread 2021年度论坛之星TOP4]、[华为云云享专家(嵌入式物联网架构设计师)]等荣誉。坚信【知识改变命运,技术改变世界】!

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 单片机
    +关注

    关注

    5922

    文章

    41499

    浏览量

    614155
  • 函数
    +关注

    关注

    3

    文章

    3248

    浏览量

    60672
  • Printf
    +关注

    关注

    0

    文章

    71

    浏览量

    13315
收藏 人收藏

    评论

    相关推荐

    有哪几种办法可实现单片机像在pc终端一样打印log

    实现单片机像在pc终端一样打印log呢?stm32实现printf打印log的办法有哪些呢?
    发表于 12-01 06:39

    单片机如何控制调试信息输出

    单片机调试log信息输出控制方法因子最近在研发公司项目:基于无线通信的****物联网应用。过程涉及到了多个程序模块,包括MQTT、JSON解析、二维码生成等。在单片机程序的编写过程,串口的调试
    发表于 01-27 07:18

    单片机是如何实现printf打印到串口的

    printf 很好用,到了单片机时却不能用,那因为 printf 在库驱动是答应到显示屏的,所用我们把驱动重新定向打印到串口就可以在单片机中用了,这样我们就可以通过串口打印到串口助手
    发表于 02-16 07:10

    如何在Keil中使用51单片机进行printf串口输出

    单片机的学习过程困扰如何使用单片机串口在上位串口助手上输出显示的问题,其实很多开发环境都支持使用C语言里面最常用的printf函数,Keil也不例外,不仅仅是最常用的STC89C52、 keil下其他51系列单片机如STC12
    发表于 08-13 17:32 14次下载
    如何在Keil中使用51<b>单片机</b>进行<b>printf</b>串口<b>输出</b>

    如何使用单片机系统重写printf函数

    单片机系统重写printf函数。
    发表于 07-23 17:37 1次下载
    如何使用<b>单片机</b>系统重写<b>printf</b>函数

    单片机printf的移植过程资料免费下载

    单片机的,说的是在51单片机编译软件库中有自带printf函数,也有自己实现一个自己的printf函数的,但是讲的不详细,看了好几个博客,东拼西凑,也在自己的单片机上弄出来了。
    发表于 07-19 17:37 2次下载
    <b>单片机</b><b>printf</b>的移植过程资料免费下载

    单片机printf重定向的函数免费下载

    单片机printf重定向的函数免费下载。
    发表于 07-03 17:42 0次下载
    <b>单片机</b><b>printf</b>重定向的函数免费下载

    AVR单片机使用printf的方法

    单片机中使用printf函数,首先得包含头文件&lt;stdio.h&gt;,其次,得在函数定义put...
    发表于 11-14 20:36 8次下载
    AVR<b>单片机</b>使用<b>printf</b>的方法

    单片机printf函数的重映射

    单片机printf函数的重映射一、源自于:大侠有话说1.如果你在学习单片机之前学过C语言,那么一定知道printf这个函数.它最最好用的功能除了打印你想要的字符到屏幕上外,还能把数字进行格式化
    发表于 11-15 20:51 11次下载
    <b>单片机</b><b>中</b><b>printf</b>函数的重映射

    51单片机串口通信调试printf函数重定向输出打印

    单片机串口通信以及printf调试串行通信1.串口初始化串口工作方式波特率设置通过软件直接生成串口初始化代码2.输出到串口3.发送字符串到串口4.printf重定向到串口5.补充
    发表于 11-20 16:36 16次下载
    51<b>单片机</b>串口通信调试<b>printf</b>函数重定向<b>输出</b>打印

    单片机控制调试信息输出的方法

    单片机调试log信息输出控制方法因子最近在研发公司项目:基于无线通信的****物联网应用。过程涉及到了多个程序模块,包括MQTT、JSON解析、二维码生成等。在单片机程序的编写过程,串口的调试
    发表于 12-03 12:06 0次下载
    <b>单片机</b>控制调试信息<b>输出</b>的方法

    单片机实现 printf 打印输出,和电脑端一样用

    printf 很好用,到了单片机时却不能用,那因为 printf 在库驱动是答应到显示屏的,所用我们把驱动重新定向打印到串口就可以在单片机中用了,这样我们就可以通过串口打印到串口助手
    发表于 12-17 18:32 1次下载
    <b>单片机</b><b>实现</b> <b>printf</b> 打印<b>输出</b>,和电脑端一样用

    stm32单片机串口使用printf及u3_printf

    单片机连接的时候使用的是串口(通常物联网用到的ESP8266,SIM9600等都是通过串口发送AT指令进行模式的配置的),但是printf函数却只有一个。在单片机printf函数默认为串口一。所以试想能不能其他串口也组成类似于printf的函数。
    发表于 12-27 19:24 1次下载
    stm32<b>单片机</b>串口使用<b>printf</b>及u3_<b>printf</b>

    基于STM8L15x单片机的串口printf输出程序分享

    单片机的串口printf输出,具体的跟随小编一起来了解一下。
    的头像 发表于 09-24 09:44 4636次阅读

    如何使用单片机printf

    printf(“”);在控制台就会输出“”的内容; 2. 通过串口
    发表于 10-29 15:48 2870次阅读
    如何使用<b>单片机</b><b>中</b><b>printf</b>