C语言,关于#运算符的宏参数转换,一道C语言的习题

今天看了一道C plus 的习题,比较困惑:

16章

题目是: 定义一个宏,以下面的格式打印名称、值和int类型变量的地址:name: fop; value: 23; address : ff464016.

答案是: #define SHOW(X) printf("name: "#X"; value:%d ; address: %p", X,&X)

我不明白 name: "#X" 这个 是什么意思? 在输出的时候,#X 怎么就能转换成 fop 字符串?

#include<stdio.h>

#define  SHOW(X)  printf("name: "#X"; value:%d ; address: %p", X,&X) 

int main(void)

{
    
    SHOW (23);
    
    return 0;
    
    
}

这个程序 在编译的时候 为什么是错的?谢谢!

'#' 的讲解:

  1. 在一个预处理器宏中的参数前面使用一个#,预处理器会把这个参数转换为一个字符数组。
  2. #是“字符串化”的意思。出现在宏定义中的#是把跟在后面的参数转换成一个字符串
  3. 其他相关 -- 单独的一个 # : 至于单独一个#,则表示对这个变量替换后,再加双引号引起来。
  4. #(stringizing)字符串化操作符。其作用是:将宏定义中的传入参数名转换成用一对双引号括起来参数名字符串。其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。
    可以参考:
    https://blog.csdn.net/xiaozhidian/article/details/119457949

编译出错:
通过宏函数可以看到,其需要输出参数的地址,但是你使用 SHOW (23); 直接使用23调用,参数是一个数字并不是变量,所以在宏函数那里&X对参数X取地址会报错。
解决方案:

#include<stdio.h>
 
#define  SHOW(X)  printf("name: "#X"; value:%d ; address: %p", X,&X) 
 
int main(void)
 
{
    int fop=23;
    SHOW (fop);
    
    return 0;
}

输出的fop是变量名,你要是用别的变量名,输出会相应改变

https://en.cppreference.com/w/cpp/preprocessor/replace#.23_and_.23.23_operators

#include <stdio.h>

#define SHOW(X) printf("name: " #X "; value:%d ; address: %p", X, &X)

int main(void)
{

    // SHOW(23); // 这里报错是因为整型常量23不能被取地址,可以改为如下形式
    int a = 23;
    SHOW(a);     // 在这里宏扩展会把#X替换为字符串"a"。在类函数宏中,#操作符会把后面的标识符经过宏扩展之后得到内容转换为字符串
    return 0;
}

#include <stdio.h>
#include<stdlib.h>
//name: fop; value: 23; address : ff464016

#define SHOW(X) printf("name: "#X"; value:%d ; address: %p", X, &X)
int main(void){
    int fop = 23;
    SHOW(fop);
    system("pause");
}

img

#X是取变量X的名字,你写的X是23,并不是变量,也没有变量名

#include<stdio.h>
 
#define  SHOW(X)  printf("name: "#X"; value:%d ; address: %p", X,&X) 
 
int main(void)
 
{

    int fop=23;
    
    SHOW (fop);
    
    return 0;
    
}

具体可参考:https://blog.csdn.net/qq_39987383/article/details/106525465