C语言##__VA_ARGS__的用法

0 前言

在调试过程中,有时会用到自定义的打印,这是就会用到##__VA_ARGS__,接下来详细讲解。

1 __VA_ARGS__的用法

自定义打印时,用到可变参数,用...即可表示可变参数,如下:

#include <stdio.h>

#define LOG1(...)               printf(__VA_ARGS__)//...表示可变参数,__VA_ARGS__就是将...的值复制到这里
int main(int argc, char** argv)
{
    char *str = "test __VA_ARGS__";
    int num = 10086;
    LOG1("this is test __VA_ARGS__\r\n");
    LOG1("this is test __VA_ARGS__:%s, %d\r\n", str, num);

    return 0;
}

打印结果如下:

this is test __VA_ARGS__
this is test __VA_ARGS__:test __VA_ARGS__, 10086

2 ##__VA_ARGS__的用法

##__VA_ARGS__前面加上##的作用是:当可变参数的个数为0时,这里的##可以把把前面多余的","去掉,否则会编译出错。
当想要在自定义的调试信息加上时间、行数等信息时,应该怎么做呢?先把正确的用法写在前面:

#include <stdio.h>

#define LOG3(fmt, ...)          printf("<%s:%s>:"fmt"\r\n", __FILE__, __FUNCTION__, ##__VA_ARGS__)
int main(int argc, char** argv)
{
    char *str = "test __VA_ARGS__";
    int num = 10086;
    
    LOG3("this is test __VA_ARGS__");
    LOG3("this is test __VA_ARGS__:%s, %d", str, num);
    LOG3();

    return 0;
}

打印如下:

<main.c:main>:this is test __VA_ARGS__
<main.c:main>:this is test __VA_ARGS__:test __VA_ARGS__, 10086
<main.c:main>:

然后做比较:如果用__VA_ARGS__,当使用时,参数为空,编译报错。如下:

#include <stdio.h>

#define LOG2(fmt, ...)          printf("<%s:%s>:"fmt"\r\n", __FILE__, __FUNCTION__, __VA_ARGS__)
int main(int argc, char** argv)
{
    LOG2();//不传参数,编译报错
    LOG2("this is test __VA_ARGS__");//编译报错
    LOG2("this is test __VA_ARGS__:%s, %d", str, num);//正常编译
    return 0;
}

编译后报错:

main.c: In function ‘main’:
main.c:3:96: error: expected expression before ‘)’ token
   3  | #define LOG2(fmt, ...)          printf("<%s:%s>:"fmt"\r\n", __FILE__, __FUNCTION__, __VA_ARGS__)
      |                                                                                                ^
main.c:6:5: note: in expansion of macro ‘LOG2’
   6  |     LOG2();//不传参数,编译报错
      |     ^~~~
main.c:3:96: error: expected expression before ‘)’ token
   3  | #define LOG2(fmt, ...)          printf("<%s:%s>:"fmt"\r\n", __FILE__, __FUNCTION__, __VA_ARGS__)
      |                                                                                                ^
main.c:7:5: note: in expansion of macro ‘LOG2’
   7  |     LOG2("this is test __VA_ARGS__");
      |     ^~~~

分析:

  • 第6行没有传参数,宏定义LOG2(fmt,...)展开后,__VA_ARGS__是空的,这时printf后面剩余一个,,必然编译失败;
  • 第7行虽然传入了字符串,但是该字符串赋值给了LOG2(fmt,...)的第一个参数fmt,宏展开后,printf后面也剩余一个,,所以编译报错。

总结

##__VA_ARGS__##的作用就是去掉前面多余的,,在使用自定义打印的时候,推荐使用第2小节的方式。

欢迎大家评论区留言交流!

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

fengwang0301

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

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

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

打赏作者

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

抵扣说明:

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

余额充值