宏定义中的特殊参数(#、##、...和__VA_ARGS__)

最近在android的某个代码的头文件中发现很多__VA_ARGS__,google一下.还是比较有用.附带其它宏定义参数,一起记录之.

1.    ...和__VA_ARGS__

看看msdn上给得例子吧

[cpp]  view plain copy
  1. #include <stdio.h> 
  2. #define EMPTY 
  3.  
  4. #define CHECK1(x, ...) if (!(x)) { printf(__VA_ARGS__); } 
  5. #define CHECK2(x, ...) if ((x)) { printf(__VA_ARGS__); } 
  6. #define CHECK3(...) { printf(__VA_ARGS__); } 
  7. #define MACRO(s, ...) printf(s, __VA_ARGS__) 
  8.  
  9. int main() { 
  10.     CHECK1(0, "here %s %s %s""are""some","varargs1(1)\n"); 
  11.     CHECK1(1, "here %s %s %s","are","some","varargs1(2)\n");  // won't print 
  12.      
  13.     CHECK2(0, "here %s %s %s","are","some","varargs2(3)\n");  // won't print 
  14.     CHECK2(1, "here %s %s %s""are""some","varargs2(4)\n"); 
  15.      
  16.     // always invokes printf in the macro 
  17.     CHECK3("here %s %s %s","are","some","varargs3(5)\n"); 
  18.      
  19.     MACRO("hello, world\n"); 
  20.     // MACRO("error\n", EMPTY);   would cause C2059 
[cpp]  view plain  copy
  1. #include <stdio.h>  
  2. #define EMPTY  
  3.   
  4. #define CHECK1(x, ...) if (!(x)) { printf(__VA_ARGS__); }  
  5. #define CHECK2(x, ...) if ((x)) { printf(__VA_ARGS__); }  
  6. #define CHECK3(...) { printf(__VA_ARGS__); }  
  7. #define MACRO(s, ...) printf(s, __VA_ARGS__)  
  8.   
  9. int main() {  
  10.     CHECK1(0, "here %s %s %s""are""some""varargs1(1)\n");  
  11.     CHECK1(1, "here %s %s %s""are""some""varargs1(2)\n");   // won't print  
  12.       
  13.     CHECK2(0, "here %s %s %s""are""some""varargs2(3)\n");   // won't print  
  14.     CHECK2(1, "here %s %s %s""are""some""varargs2(4)\n");  
  15.       
  16.     // always invokes printf in the macro  
  17.     CHECK3("here %s %s %s""are""some""varargs3(5)\n");  
  18.       
  19.     MACRO("hello, world\n");  
  20.     // MACRO("error\n", EMPTY);   would cause C2059  
  21. }  

我们看看输出:

here are some varargs1(1)
here are some varargs2(4)
here are some varargs3(5)
hello, world

是不是很有趣,以后我们如果在一些情况下要定义自己的打印函数就特别方便了

比如我们要在A条件成立,且B条件不成立的情况下打印输出

可以做如下定义:

[cpp]  view plain copy
  1. #define CHECK1(A,B,...) if ((A)&&!(B)) { printf(__VA_ARGS__); } 
[cpp]  view plain  copy
  1. #define CHECK1(A,B,...) if ((A)&&!(B)) { printf(__VA_ARGS__); }  

记住一点...只能代替后面的参数,如下定义可行不通:

[cpp]  view plain copy
  1. #define W(x,...,y) 
[cpp]  view plain  copy
  1. #define W(x,...,y)  

是不是很像C++的函数的默认参数一样.


2.   #

这个特殊的宏定义参数也特别有用.#作为一个预处理运算符,它可以把语言符号字符串化(stringizing).例如我们定义的变量等.

看一个简单的例子:

[cpp]  view plain copy
  1. #include <stdio.h> 
  2. #define TEST(x) printf("square of  " #x " is %d.\n",(x)*(x)) 
  3. void main() 
  4.     int y =4; 
  5.     TEST(y); 
  6.     TEST(6-3); 
  7.     TEST(y+3); 
[cpp]  view plain  copy
  1. #include <stdio.h>  
  2. #define TEST(x) printf("square of  " #x " is %d.\n",(x)*(x))  
  3. void main()  
  4. {  
  5.     int y =4;  
  6.     TEST(y);  
  7.     TEST(6-3);  
  8.     TEST(y+3);  
  9. }  

输出结果:

[cpp]  view plain copy
  1. square of  y is 16. 
  2. square of  6-3 is 9. 
  3. square of  y+3 is 49. 
[cpp]  view plain  copy
  1. square of  y is 16.  
  2. square of  6-3 is 9.  
  3. square of  y+3 is 49.  

3.   ##

##运算符可以用于类函数宏的替换部分.##还可以用于类对象宏的替换部分.这个运算符可以把两个语言符号组合成单个语言符号.

例如:

  1. #define XNAME(n) x##n 
  2. 执行宏调用: 
  3. int XNAME(4) 
  4. 展开后: 
  5. x4;//相当于直接定义 int x4 这里也就体现了两个语言符号的拼接.不过在android经常看到有关##拼接语言符号的情况. 

 

守护进程则可以改为syslog等等... 其中,决窍其实就是这几个宏 ##__VA_ARGS__, __FILE__, __LINE__ 和__FUNCTION__,下面介绍一下这几个宏:
  1) __VA_ARGS__ 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。宏前面加上##的作用在于,当可变参数的个数为0时,这里的##起到把前面多余的","去掉的作用,否则会编译出错, 你可以试试。
  2) __FILE__ 宏在预编译时会替换成当前的源文件名
  3) __LINE__宏在预编译时会替换成当前的行号
  4) __FUNCTION__宏在预编译时会替换成当前的函数名称

 

3.可变参数宏 ...和_ _VA_ARGS_ _

其中,...表示参数可变,          __VA_ARGS__定义好的宏

__VA_ARGS__ 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。这样预定义宏_ _VA_ARGS_ _就可以被用在替换部分中,替换省略号所代表的字符串。比如:
#define PR(...) printf(__VA_ARGS__)
int main()
{
    int wt=1,sp=2;
    PR("hello\n");
    PR("weight = %d, shipping = %d",wt,sp);
    return 0;
}
省略号只能代替最后面的宏参数。
#define W(x,...,y)错误!

 

飞行棋例子:

#define CHARFMT(__format__,...) String::createWithFormat(__format__,##__VA_ARGS__)->getCString()

#define DYNAMIC_CAST_READER(__type__,__path__) dynamic_cast<__type__*>(cocostudio::GUIReader::getInstance()->widgetFromJsonFile(__path__));
#define OBJ2OBJ_STATIC_CAST(__ccobject__,__type__) static_cast<__type__*>(__ccobject__)
#define OBJ2OBJ_DYNAMIC_CAST(__ccobject__,__type__) dynamic_cast<__type__*>(__ccobject__)
#define AUDIO_PLAYBACKGROUND(__format__) MusicManager::getInstance()->playBackgroundMusic(__format__);
#define AUDIO_PLAYEFFECT(__format__) MusicManager::getInstance()->playEffectMusic(__format__);

4 常量及宏的命名

采用下划线分割大写字母的方式命名,一般应以设备名作为前缀,
防止模块间命名的重复。如:

#define TIMER0_MODE_RELOAD 2
#define TIMER2_COUNT_RETRIEVE(val) ((uint16)(65536 - (val)))

当然,看作接口的宏可以按照函数的命名方法命名,例如:

#define timer2_clear() (TF2 = 0)
#define timer0_is_expired() (TF0)
----------------------

#define CHARFMT(__format__,...) String::createWithFormat(__format__,##__VA_ARGS__)->getCString()

这句话使用下面更具有健状性

#define CONVERT_TO_INT(__format__ , ...)\
  do {\
  if (__format__)\
  {\
  CCLOG(__VA_ARGS__);\
  }\
  } while (0);

  CONVERT_TO_INT(10, "herr is %s %s", "hello", "are");

注: 1  第一个 ##不能去掉 ?  第二个为何不用加 ##

2 测试发现  第一个参数__format__ 多余  weisha?

3  名字一定要大写 

4  \  之前不能有空格  老王

 

[cpp]  view plain  copy
  1. #define XNAME(n) x##n  
  2. 执行宏调用:  
  3. int XNAME(4)  
  4. 展开后:  
  5. x4;//相当于直接定义 int x4 这里也就体现了两个语言符号的拼接.不过在android经常看到有关##拼接语言符号的情况.
引用:建议使用__VA_ARGS__时与##配合使用;##__VA_ARGS__在设置log输出的宏定义时比较常见。 引用:程序开发或者看工程代码时可能在宏定义有涉及到#、##、__VA_ARGS__和##__VA_ARGS__的使用,简单介绍下使用方法。#操作符是将其后面紧跟的符号变为字符串的形式。 #__VA_ARGS__是C语言宏定义的一个特殊标识符,表示可变参数列表。它可以用来在宏定义传递不定数量的参数。当我们在宏定义使用#__VA_ARGS__时,它会被替换为传递给宏的所有参数,以逗号分隔的形式。这样,我们可以在宏定义使用这些参数,类似于函数的参数。 而##__VA_ARGS__是在宏定义使用可变参数列表时常见的形式。它的作用是在宏定义使用可变参数列表时,可以将参数列表前的逗号省略掉。这样可以避免在宏定义使用可变参数列表时出现额外的逗号。 所以,#__VA_ARGS__和##__VA_ARGS__都是在宏定义使用可变参数列表时的常见用法,可以实现灵活的宏定义,提高代码的可读性和可维护性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [# ## __VA_ARGS__ ##__VA_ARGS__的使用说明](https://blog.csdn.net/hanxv_1987/article/details/106555870)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值