重载内置(固有?)功能
Overload built-in (intrinsic?) function
考虑以下代码:
#include <iostream>
#include <math.h>
double log(double) { return 42; }
int main() {
std::cout << log(1) << std::endl;
}
在构建调试版本时,所有使用的编译器(msvc、gcc、clang)都会打印42
。
但当我尝试在发布模式下构建(并运行)时,我得到了:
- msvc:
error C2169: 'log' : intrinsic function, cannot be defined
中的编译错误 - 为gcc打印CCD_ 3
- 为clang打印
0
为什么同一编译器的发布/调试结果不同?
为什么不同的编译器在发布模式下会得到不同的结果?
您正在定义一个已在<math.h>
中通过外部链接声明的函数。
C11标准,§7.12.6.7:
#include <math.h> double log(double x);
§7.1.2:
库函数的任何声明都应具有外部链接。
[外部名称]/3:
用外部链接声明的标准C库中的每个名称保留给实现,用作外部为"C"的名称链接,位于命名空间
std
和全局命名空间中。
根据[reserved.names]/2,行为是未定义的;因此,实现可以随心所欲,包括发布无意义的错误消息。
因此,根据标准(17.6.1.2.4):
然而,在C++标准库中,声明(在C中定义为宏的名称除外)在命名空间std的命名空间范围(3.3.6)内。未指定这些名称是否首先在全局命名空间范围内声明,然后通过显式使用声明(7.3.3)注入命名空间std
未指定math.h
中的log()
(实际上是cmath
)是否在命名空间std
中。如果是(就像对gcc的libstdc++一样),那么调用log(1)
非常简单地调用您的函数,因为另一个函数名为std::log()
。但对于clang,显然它将其放在了全局命名空间中。既然有
template <typename T> double log(T x);
这将比你的更可取,因为你正在传递一个int,所以当clang时,它会调用这个。(我现在无法检查,因为我无法访问coliru,也没有安装clang,但这是最好的猜测)。
相关文章:
- 在执行其他功能的同时播放动画(LED矩阵和Arduino/ESP8266)
- 多态性和功能结合
- 带内存和隔离功能的SQLite
- 在CMakeLists.txt的安装功能中使用.cmake文件有什么用
- 类模板的成员功能的定义在单独的TU中完全专业化
- 有没有一种方法可以创建一个带有哈希表的数据库,该哈希表具有恒定时间查找功能
- 如何在C++中获得"静态纯虚拟"功能?
- 两个文件使用彼此的功能-如何解决
- 我应该实现右值推送功能吗?我应该使用std::move吗
- QML按钮点击功能执行顺序
- 无法理解此 return 语句的功能,没有它就会发生运行时错误
- 有没有可能有一个只有ADL才能找到的非好友功能
- 功能样式转换从 'int' 到 'ItemType' 的匹配转换
- 文件系统:复制功能的速度秘诀是什么
- 在用于格式4的arm模拟器中实现功能时的一个问题
- 如何在Directwrite中获得给定字体的可用OpenType功能
- 对可变参数使用声明.如何选择正确的功能
- 由于Memcpy的问题,无法使用VS进行优化;被视为固有功能
- 对 64 位整数中的尾随零位进行计数的固有功能
- 重载内置(固有?)功能