考虑以下程序。
实践
#include <stdio.h>
#define P(x,y,z) (x+y+z)
main()
{
int x,y,z,i,j,k;
printf("\n Number I = ");
scanf("%d",&i);
printf("\n Number II = ");
scanf("%d",&j);
printf("\nNumber III = ");
scanf("%d",&k);
printf("\n Result = %d\n",P(i,j,k));
}
现在,如果我运行
gcc -E Practice.c
,输出将是# 2 "Practice.c" 2
main()
{
int x,y,z,i,j,k;
printf("\n Number I = ");
scanf("%d",&i);
printf("\n Number II = ");
scanf("%d",&j);
printf("\nNumber III = ");
scanf("%d",&k);
printf("\n Result = %d\n",(i+j+k));
}
在这里,宏
P(i,j,k)
被(i+j+k)
替换,并且在命令屏幕上清晰可见。我的问题是,这里的
#include <stdio.h>
到底发生了什么?它是否也被
stdio
的文件替换,与#define P(x,y,z) (x+y+z)
相同,还是只是将stdio
的内容链接到main()
的一种方式。而且,使用
# 2 "Practice.c" 2
后输出中的gcc -E Practice.c
是什么? 最佳答案
在预处理期间,#include指令实际上被包含文件的内容替换。此外,此包含文件本身已经过预处理,因此,如果包含任何其他包含指令,则将包含另一个文件,依此类推。
通常,预处理器和编译器是两个不同的程序,预处理的输出通过管道发送给编译器。编译器本身不会打开和读取源文件。
在这种情况下,很明显,必须存在一种机制,预处理器如何才能告诉编译器代码的来源。对于错误消息,这是必需的。如果存在语法错误,则预处理器不会检测到该语法错误,编译器会发现该语法错误,并且编译器需要显示错误所在的确切行和源文件。但是,当仅从管道读取一个流时,如果没有其他形式就无法获得此信息。这就是为什么线像
# 2 "Practice.c" 2
出现。它是有关源代码实际来源的信息。在这种情况下,预处理器通知此行之后的代码来自文件“ Practice.c”,而下一行是此源代码的第二行。然后,编译器可以增加下一行代码的行号,直到下一个这样的指令(通常被调用并用作#line指令)为止。
关于c - 宏模板和扩展,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/23588791/