标准目录中包含的文件可以静默替换现有宏
Included files from standard directories can silently replace existing macros?
使用单行创建文件/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
找到的另一个目录中不会产生这种效果。
系统头文件通常不能用严格符合 C 格式编写。它们可能会多次更改某些预处理器宏,如以下示例所示:
#define FOO 0
#ifdef BAR
#define FOO 1
#endif
对所有这些更改发出警告会很烦人。因此,不会对系统头文件发出警告。
您的代码行
#include <define_x.h>
将头文件声明为系统头文件。如果您使用
#include "define_x.h"
相关文章:
- 模板参数替换失败,并且未完成隐式转换
- 如何用转义符替换字符串中的所有特殊字符
- 为什么除非添加括号,否则构造函数上的模板替换会失败?
- 在一个读写器队列中,我可以用volatile替换原子吗
- 用符号版本替换对函数的所有调用
- 如何通过替换顺序代码的while循环来添加OpenMP for循环
- 替换基于地图的所有引用
- 按平均值替换数组中的元素
- 我可以在这里替换什么,因为我不能在 C# 中使用隐式变量的 lambda 函数?
- 如何将字节数组元素替换为修改的十六进制 ASCII 符号?
- 初始化 std::vector 替换为单大括号而不是双大括号
- 删除/替换C++字符串中的多字符 (ÿû)
- 将 malloc 替换为数组
- 如何替换此示例代码片段中已弃用的handler_type_t或 boost::asio::handler_type?
- 如何在C++中用'\'替换''来处理转义序列?
- 替换密码:哪一个?
- 替换 C++17 中移除的绑定 1st
- 将 Boost MPL 容器替换为 C++17 功能
- C++:如何用单个命令替换复杂的迭代?
- 标准目录中包含的文件可以静默替换现有宏