标准目录中包含的文件可以静默替换现有的宏吗?

时间:2019-04-16 16:58:26

标签: c++ preprocessor

用单行创建文件/usr/local/include/define_x.h

#define X "/usr/local/include"

以下程序foo.cpp编译并运行正常,没有警告:

#include <iostream>

#define X "local"
#include <define_x.h>

int main()
{
  std::cout << "X = " << X << std::endl;
  return 0;
}

输出:X = /usr/local/include

现在交换第3行和第4行:

#include <iostream>

#include <define_x.h>
#define X "local"

int main()
{
  std::cout << "X = " << X << std::endl;
  return 0;
}

现在输出为预期的X = local,但是现在出现编译器警告:

foo.cpp:4: warning: "X" redefined
 #define X "local"

In file included from foo.cpp:3:
/usr/local/include/define_x.h:1: note: this is the location of the previous definition
 #define X "/usr/local/include"

问题:为什么第一版中没有警告?

两者均使用gcc 8.2.1:g++ foo.cpp编译。

似乎与将文件放置在标准/ usr / include或/ usr / local / include目录中有关,因为将它们放置在当前目录或通过-I找到的另一个目录中不会产生这种效果。

1 个答案:

答案 0 :(得分:1)

System header files通常不能用严格符合C的语言编写。它们可能会多次更改某些预处理器宏,例如以下示例:

#define FOO 0
#ifdef BAR
#define FOO 1
#endif

对所有这些更改发出警告会很烦人。因此,不会对系统头文件发出警告。

您的代码行

#include <define_x.h>

将头文件声明为system header file。如果使用

,应该会得到不同的结果
#include "define_x.h"