C语言宏#define中#,##,#@和\的用法

一、(#)字符串化操作符

作用:将宏定义中的传入参数名转换成用一对双引号括起来参数名字符串。其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。

如:

#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"

二、 (##)符号连接操作符

作用:将宏定义的多个形参转换成一个实际参数名。
如:

#define exampleNum( n )  num##n

使用:

int num9 = 9;
int num = exampleNum( 9 ); // 将会扩展成 int num = num9

注意:

a. 当用##连接形参时,##前后的空格可有可无。

如:  #define exampleNum( n )       num ## n                 
// 相当于 #define exampleNum( n )      num##n

b. 连接后的实际参数名,必须为实际存在的参数名或是编译器已知的宏定义。

c. 如果##后的参数本身也是一个宏的话,##会阻止这个宏的展开。

#include <stdio.h>
#include <string.h>

#define STRCPY(a, b)   strcpy(a ## _p, #b)
int main()
{
    char var1_p[20];
    char var2_p[30];
    strcpy(var1_p, "aaaa");
    strcpy(var2_p, "bbbb");
    STRCPY(var1, var2);
    STRCPY(var2, var1);
    printf("var1 = %s\n", var1_p);
    printf("var2 = %s\n", var2_p);

    //STRCPY(STRCPY(var1,var2),var2);
    //这里是否会展开为: strcpy(strcpy(var1_p,"var2")_p,"var2“)?答案是否定的:
    //展开结果将是:  strcpy(STRCPY(var1,var2)_p,"var2")
    //## 阻止了参数的宏展开!如果宏定义里没有用到 # 和 ##, 宏将会完全展开
    // 把注释打开的话,会报错:implicit declaration of function 'STRCPY'
    return 0;
}  

结果:

var1 = var2
var2 = var1

三、 (#@)单字符化操作符

作用:将传入单字符参数名转换成字符,以一对单引用括起来。

如:

#define makechar(x)    #@x
char a = makechar( b ); //展开后变成了:a = 'b';

四、(\) 续行操作符

作用:当定义的宏不能用一行表达完整时,可以用”\”表示下一行继续此宏的定义。

注意 \ 前留空格。

当宏参数是另一个宏的时候,需要注意的是凡宏定义里有用’#’或’##’的地方宏参数是不会再展开。

1、非#和##的情况

#define TOW       (2) 
#define MUL(a,b) (a*b) 
printf("%d*%d=%d\n", TOW, TOW, MUL(TOW,TOW)); 
//这行的宏会被展开为: 
printf("%d*%d=%d\n", (2), (2), ((2)*(2))); 
//MUL里的参数TOW会被展开为(2). 

2、当有#或##的时候

#define A           2
#define STR(s)      #s 
#define CONS(a,b)   (int)a##e##b
printf("int max: %s\n",   STR(INT_MAX));     // INT_MAX #include <limits>
//这行会被展开为: 
//printf("int max: %s\n", "INT_MAX"); 
printf("%s\n", CONS(A, A));                // compile error  
//这一行则是: 
//printf("%s\n", int(AeA)); 
//INT_MAX和A都不会再被展开, 然而解决这个问题的方法很简单. 加多一层中间转换宏.
//加这层宏的用意是把所有宏的参数在这层里全部展开, 那么在转换宏里的那一个宏(_STR)就能得到正确的宏参数. 
#define A            2
#define _STR(s)      #s 
#define STR(s)       _STR(s)           // 转换宏 
#define _CONS(a,b)   (int)a##e##b 
#define CONS(a,b)    _CONS(a,b)        // 转换宏 
printf("int max: %s\n", STR(INT_MAX)); // INT_MAX,int型的最大值,#include <limits>
//输出为: int max: 2147483647 
//STR(INT_MAX) -->   _STR(2147483647) 然后再转换成字符串; 
printf("%d\n", CONS(A, A)); 
//输出为:200 
//CONS(A, A)  -->  _CONS(2, 2) --> (int)2e2 

参考:https://blog.csdn.net/bytxl/article/details/47423151
https://blog.csdn.net/roger_ranger/article/details/78305021

  • 35
    点赞
  • 101
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
C语言,#define是用来定义常量和的预处理指令。常见的用法包括两种,一种是定义常量,另一种是定义带参数的。#define定义的常量也称为符号常量,其值在程序运行过程不能被改变。常量的定义形式为#define 值。而定义带参数的则使用#define 名(参数列表) 表达式的形式,名后面的参数列表用于接收传入的参数,并在的展开过程进行替换。通过使用#define定义,可以像函数一样解决问题,并且可以将参数替换到文本。例如,可以使用定义数组的大小,如#define MAX 1000,然后在代码使用该来定义数组的大小,如int arr[MAX]。这样可以方便地统一修改数组大小而不需要逐个修改相关的代码。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [C语言之#define用法入门详解](https://blog.csdn.net/sunnyoldman001/article/details/127895225)[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_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [【C语言】预处理详解:#define的各种使用方法](https://blog.csdn.net/wangduduniubi/article/details/129947832)[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_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值