宏定义中的##

关于宏定义中#的用法,下面两篇帖子可以参考:

  1. https://blog.csdn.net/xdsoft365/article/details/5911596
  2. https://blog.csdn.net/u013187074/article/details/78874976
    其中第2篇是单独介绍可变参数arg…的宏定义的,主要用于log系统等,比如
#ifndef BT_ERROR
#define BT_ERROR(module_id, error_code, format, arg...) \
  am_bt_error(__PRETTY_FUNCTION__, __FILE__, __LINE__, module_id, error_code, format, ##arg)
#endif

上面例子中,##arg就是为了代表arg…的。

2、#define D(a) cout << #a “=[” << a << “]” << endl;

3、#是“字符串化”的意思。出现在宏定义中的#是把跟在后面的参数转换成一个字符串

例如:
define FOO(arg) my##arg

FOO(abc)
相当于 myabc
例如:
define STRCPY(dst, src) strcpy(dst, #src)

STRCPY(buff, abc)
相当于 strcpy(buff, “abc”)
另外,如果##后的参数本身也是一个宏的话,##会阻止这个宏的展开,也就是只替换一次。
define STRCPY(a, b) strcpy(a ## _p, #b)
int main()
{
char var1_p[20];
char var2_p[30];
/* 注意这里 /
STRCPY(STRCPY(var1,var2),var2);
/
这里是否会展开为: strcpy(strcpy(var1_p,“var2”)_p,"var2“)?
* 答案是否定的:
* 展开结果将是: strcpy(STRCPY(var1,var2)p,“var2”)
* ## 阻止了参数的宏展开!
* 如果宏定义里没有用到 # 和 ##, 宏将会完全展开
*/
}
http://blog.chinaunix.net/u/17855/showart_113663.html
4、about ## in common text
(1)关于记号粘贴操作符(token paste operator): ##
简单的说,“##”是一种分隔连接方式,它的作用是先分隔,然后进行强制连接。
其中,分隔的作用类似于空格。我们知道在普通的宏定义中,预处理器一般把空格解释成分段标志,对于每一段和前面比较,相同的就被替换。但是这样做的结果是,被替换段之间存在一些空格。如果我们不希望出现这些空格,就可以通过添加一些##来替代空格。
另外一些分隔标志是,包括操作符,比如 +, -, *, /, [,], …,所以尽管下面的
宏定义没有空格,但是依然表达有意义的定义: define add(a, b) a+b
而其强制连接的作用是,去掉和前面的字符串之间的空格,而把两者连接起来。
(2)举列 – 试比较下述几个宏定义的区别
define A1(name, type) type name
##type##type 或
define A2(name, type) type name##
##type##_type
A1(a1, int); /* 等价于: int name_int_type; /
A2(a1, int); /
等价于: int a1_int_type; */
解释:

  1. 在第一个宏定义中,“name"和第一个”“之间,以及第2个”"和第二个"type"之间没有被分隔,所以预处理器会把name_##type##type解释成3段:
    “name
    ”、“type”、以及“_type”,这中间只有“type”是在宏前面出现过
    的,所以它可以被宏替换。
  2. 而在第二个宏定义中,“name”和第一个“”之间也被分隔了,所以
    预处理器会把name##
    ##type##type解释成4段:“name”、“”、“type”
    以及“_type”,这其间,就有两个可以被宏替换了。
  3. A1和A2的定义也可以如下:
    define A1(name, type) type name_ ##type ##type
    <##前面随意加上一些空格>
    define A2(name, type) type name ##
    ##type ##_type
    结果是## 会把前面的空格去掉完成强连接,得到和上面结果相同的宏定义
    (3)其他相关 – 单独的一个 #
    至于单独一个#,则表示对这个变量替换后,再加双引号引起来。比如
    define __stringify_1(x) #x
    那么
    __stringify_1(linux) <==> “linux”
    (5)#(stringizing)字符串化操作符。其作用是:将宏定义中的传入参数名转换成用一对双引号括起来参数名字符串。其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。
    如:
    define example(instr) printf(“the input string is:\t%s\n”,#instr)
    define example1(instr) #instr
    当使用该宏定义时:
    example(abc); 在编译时将会展开成:printf(“the input string is:\t%s\n”,“abc”);
    string str=example1(abc); 将会展成:string str=“abc”;
    注意:
    对空格的处理
    a。忽略传入参数名前面和后面的空格。
    如:str=example1( abc ); 将会被扩展成 str=“abc”;
    b.当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串中只以一个空格连接,忽略其中多于一个的空格。
    如:str=exapme( abc def); 将会被扩展成 str=“abc def”;
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值