C/C++中自定义信息输出——printf与宏的配合使用

在C/C++中,提起“宏”多少有些皱眉,至少我在入门C++时旁人好心提醒:尽可能地使用typedef与const常量定义来替代“宏”的使用:

1. 类型宏定义

  #define HANLE void*
	//可以替换为: 
	typedef void *HANLE;

2. 常量定义

  #define MAX_LIMIT 4096
	//可以替换为:
	const int s_MAX_LIMIT 4096;

因为宏定义是在预编译阶段,对内容进行直接替换,因此无法提供安全的类型检查等功能。

但我本次要说的是,宏的一个很炫的应用:

配合printf函数能够方便地自定义输出格式和内容

举例来说吧,就是用户在程序中调用:

USER_PRINT( "i should checkout: v_a value is %d.\n", value_a );

能够输出类似信息

[2013-10-01 15:18:28][Fox Test][PID:2486]i should checkout: v_a value is 10.

当然,如果你用过linux下的syslog,你会发现这条输出信息算是“高仿”了。syslog可是linux下调试/监控的优秀组件,简单易用不说,功能也比较完备!

扯远了,本次的目标就是如何在用户简单输出一条信息的时候,我们能够同时加上各种附加信息,包括时间/用户/PID等参数,方便我们的调试/监控。


1. 首先,提供一个有意思的宏,该宏也是该功能的核心

#define USER_PRINT_BASE( format, args... ) 	printf( format, ##args )

是不是看到args...有点儿蒙,以及##args又是闹哪样?

说明一下:

i: args...是一个参数,是C/C++中定义变参函数(参数不固定)的一种格式。至于举个变参函数的例子,printf/scanf什么的就是随处可见的例子。

ii: ##args是宏的一个特殊语法应用(和普通的##宏应用不同[TODO]),使得宏展开时可以去掉可能存在的多余的','。

举例:

当没有##时,USER_PRINT_BASE( "something debug" )在宏展开时会成为printf( "something debug", )多了一个逗号!

当然,如果你调用USER_PRINT_BASE( "something debug--%d", 10 )的时候,即使没有'##'也不会出错。

iii: 如果你的编译器支持C99规范,那么宏也可以改写为:

#define USER_PRINT_BASE( format, ... ) 		printf( format, ##__VA_ARGS__ )


嗯好吧,其实只是变参的书写格式发生了点变化。

注:ii与iii详细的可以参考“C/C++可变参数,“## __VA_ARGS__”宏的介绍和使用”,里面有很详细的介绍。

2. 之后,在核心功能之上,我们实现功能的叠加吧:

#include <time.h>
#define USER_PRINT_0( format, args... ) \
{ \
        time_t timep; \
        time( &timep ); \
        struct tm *pTM = gmtime( &timep ); \
        USER_PRINT_BASE( "[%4d-%02d-%02d %02d:%02d:%02d]"format, \
                pTM->tm_year+1900, pTM->tm_mon, pTM->tm_mday, pTM->tm_hour+8, pTM->tm_min, pTM->tm_sec, ##args ) ; \
}

这是加入时间信息。好吧,我承认时间的处理上我土了,但是asctime函数返回时默认有个换行符,待之后在订正吧。[TODO]。

需要注意:

i: 为什么这段前后要加大括号咧?利用作用域及时销毁time_t变量和pTM变量,否则连续调用USE_PRINT_0会出现重定义错误哦~

ii: USE_PRINT_0中的是(args...),在替换的USE_PRINT_BASE中还是需要(##args)哦~

iii: 附加的格式化内容也是可以放在format之后的哦,当然,注意调节变参的顺序就好。

3. 接下来,就像搭积木一样来就好了:

#include <unistd.h>
#define MAX_LIMIT 4096
const char useName[4096] = "Fox Test";
#define USER_PRINT_1( format, args... )		USER_PRINT_0( "[%s]"format, useName, ##args )
#define USER_PRINT_2( format, args... )		USER_PRINT_1( "[PID:%d]"format, getpid(), ##args )	//getpid() 获取当前进程ID
#define USER_PRINT( format, args...)		USER_PRINT_2( format, ##args )

读者可能感叹,竟然可以这样往上加参数么?我的建议是,自己玩儿一下,这的很有趣。

用户调用

直接调用类似printf一样调用USER_PRINT即可默认地输出多种调试信息,十分方便!

完整代码:

#include <stdio.h>

#include <time.h>

#include <unistd.h>

#define MAX_LIMIT 4096
const char useName[4096] = "Fox Test";

#define USER_PRINT_BASE( format, args... ) 	printf( format, ##args )

#define USER_PRINT_0( format, args... ) \
{ \
        time_t timep; \
        time( &timep ); \
        struct tm *pTM = gmtime( &timep ); \
        USER_PRINT_BASE( "[%4d-%02d-%02d %02d:%02d:%02d]"format, \
                pTM->tm_year+1900, pTM->tm_mon, pTM->tm_mday, pTM->tm_hour+8, pTM->tm_min, pTM->tm_sec, ##args ) ; \
}
#define USER_PRINT_1( format, args... )		USER_PRINT_0( "[%s]"format, useName, ##args )
#define USER_PRINT_2( format, args... )		USER_PRINT_1( "[PID:%d]"format, getpid(), ##args ) //getpid() 获取当前进程ID
#define USER_PRINT( format, args...)		USER_PRINT_2( format, ##args )

int main()
{
	USER_PRINT( "Test\n" );
	USER_PRINT( "Test %d\n", 10 );
	return 0;
}



参考资料:1. C/C++可变参数,“## __VA_ARGS__”宏的介绍和使用

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 本章主要针对初学者,简要介绍了c/c++的基本语法和常见的编译错误。首先,介绍了程序的基本结构和常用的数据类型。c语言的基本语句包括赋值语句、条件语句和循环语句等,这些语句都可以组成程序的逻辑结构。接下来,介绍了函数的定义和调用,以及参数传递的方式。对于c++,还介绍了一些面向对象的概念,例如类、对象、成员函数等。同时,也提到了头文件和命名空间的使用方法。 在编程过程,常常会出现各种编译错误,例如语法错误、类型不匹配、语义错误等,需要学会如何查看和解决这些错误。此外,还介绍了调试工具和各种常用的运算符和表达式,这些都是初学者必须掌握的基础知识。 总的来说,本章是关于c/c++快速入门的一篇简介性文章。虽然只是涉及到了基础的语法和知识点,但对于初学者而言是一个很好的起点。在学习过程需要不断实践,积累经验,并不断深入了解更高级的编程技术和工具。 ### 回答2: C/C++是一种广泛使用的编程语言,它具有高效、灵活、可移植等特点,在各个领域得到广泛应用。C语言是C++语言的基础,在学习C++之前,需要先掌握C语言的基础知识。 本文介绍的是C/C++教程的第二章——快速入门C/C++。在这一章节,我们将介绍C语言的基本语法、变量、运算符和流程控制语句等基础知识。以下是C语言的一些基本知识点。 C语言的基本语法: C语言程序由多个函数组成,其一个函数必须命名为main(),程序从该函数开始执行。C程序的语句以分号结束,注释使用“//”表示单行注释,“/* */”表示多行注释。 变量和数据类型: C语言变量的定义格式为“数据类型 变量名”;数据类型包括基本类型和用户自定义类型。C语言的基本类型有int类型、char类型、float类型和double类型等。其,int类型表示整型,char类型表示字符型,float类型和double类型表示浮点型。 运算符: C语言的运算符包括算术运算符、关系运算符、逻辑运算符等。例如,“+”表示加法运算符,“>=”表示大于等于运算符,“&&”表示逻辑与运算符。 流程控制语句: C语言的流程控制语句包括if语句、switch语句、while语句、do-while语句和for语句等。这些语句可以根据条件执行相应的语句块。 总之,本章节的快速入门C/C++,具有基本语法、变量、运算符和流程控制语句等基础知识。初学者可以通过这些基础知识,轻松入门C/C++,为后续学习打下基础。同时,要注意编写代码的规范和逻辑性,才能更好的理解和使用C/C++语言。 ### 回答3: C语言是一门广泛使用的编程语言,具有高效、灵活、稳定等特点,被广泛应用于嵌入式系统、操作系统、驱动程序、多媒体应用等领域。学习C语言是程序员的必备技能之一。 第二章的快速入门C/C++教程,主要介绍了C/C++语言的基础知识,重点是程序的结构和输入输出。其程序结构包括函数、语句、变量、表达式等,而输入输出则包括scanf、printf、getchar和putchar等函数。 # 程序结构 程序结构是指程序的基本构成单元,包括函数、语句、变量、表达式等。C语言的程序结构主要包含以下几个方面。 ## 函数 C语言,函数是程序的基本组成单元。一个C程序可以由一个或多个函数组成,每个函数可以完成一个任务。函数的格式如下: ```c 返回类型 函数名(参数1, 参数2, ...){ // 函数体 return 返回值; } ``` 其,返回类型指函数执行后的返回值类型;函数名是由程序员定义的,用于调用函数时识别函数;参数列表是函数的输入参数,可以有多个参数,每个参数由类型和变量名组成;函数体是函数要执行的代码块;return语句可以返回函数的执行结果。 ## 语句 语句是完成特定功能的一组指令。C语言的语句包括赋值语句、条件语句、循环语句等。C语言通常使用花括号来表示语句块。例如,下面是一个if语句的例子。 ```c if(条件){ // if语句块 }else{ // else语句块 } ``` 如果条件为真,则执行if语句块的代码;否则执行else语句块的代码。 ## 变量 变量是用于存储数据的一种容器。在C语言,一个变量包括变量名、类型和值。变量名由程序员定义,用于识别变量;类型指变量的数据类型,如整型、字符型、实型等;值是存储在变量的数据。变量的定义格式如下。 ```c 数据类型 变量名 = 值; ``` 例如,下面是一个整型变量的定义。 ```c int num = 10; ``` ## 表达式 表达式是由变量、运算符和常量组成的一个具有返回值的语句。C语言的运算符分为算术运算符、关系运算符、逻辑运算符等,例如加号、减号、乘号、除号等。下面是一个简单的表达式。 ```c a = 5 + 6 * 3 / 2 - 1; ``` 这个表达式将计算5加6乘3除以2减1的值,并将结果赋给a变量。 # 输入输出 输入输出是程序非常重要的部分,可以让程序与用户进行交互。C语言有多种输入输出函数,其一些最常用的是scanf、printf、getchar和putchar函数。 ## scanf函数 scanf函数用于从标准输入读取格式化数据,并将读取的数据存储到变量。它的格式如下。 ```c scanf("格式控制字符串", 变量列表); ``` 其,格式控制字符串指示scanf函数需要读取的数据类型和格式,变量列表指向要读取的变量。下面是一个scanf函数的例子。 ```c int num; scanf("%d", &num); ``` 这个代码段将从标准输入读取一个整数,并将其存储到num变量。 ## printf函数 printf函数用于将格式化数据输出到标准输出。它的格式如下。 ```c printf("格式控制字符串", 参数列表); ``` 其,格式控制字符串指示printf函数需要输出的数据类型和格式,参数列表包含要输出的变量和常量。下面是一个printf函数的例子。 ```c int num = 5; printf("num的值是%d\n", num); ``` 这个代码段将输出“num的值是5”。 ## getchar和putchar函数 getchar函数用于从标准输入读取一个字符,putchar函数用于将一个字符输出到标准输出。它们的用法非常简单,例如下面的代码将读取一个字符并将其转换成大写字母后输出。 ```c char c = getchar(); putchar(toupper(c)); ``` 以上就是第二章的快速入门C/C++教程的主要内容,包括程序结构和输入输出方面的基础知识。熟练掌握这些内容,对于学习C语言来说是非常重要的。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值