考虑以下程序。





  实践


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

10-12 16:07