手动定义的 strlens 的奇怪行为
Weird behavior with a manually defined strlen
偶然地,我写了以下有趣的片段:
#include <iostream>
#include <cstring>
size_t strlen(const char* str) {
std::cout << "hello";
return 0;
}
int main() {
return std::strlen("sdf");
}
出乎我的意料的是,GCC 5.1 中的输出是"hello",这意味着我的strlen
被调用。更有趣的是,如果我删除return
,即仅用std::strlen("sdf");
调用替换main,则不会打印任何内容!
我还尝试了 Clang,std::strlen
调用计算字符串长度的 real 函数(并且没有打印任何内容)。这就是我期望看到的。
这怎么解释呢?定义我自己的strlen
函数是否被视为未定义的行为?
这里没有什么有趣的,只是一个函数重载和一些未定义的行为。您使用自己的版本重载了库函数strlen()
。由于在GCC中std::strlen
的实现只不过是命名空间std
中的库函数调用,因此您可以得到所看到的结果。
以下是cstring
的相关摘录:
namespace std _GLIBCXX_VISIBILITY(default)
{
_GLIBCXX_BEGIN_NAMESPACE_VERSION
using ::strlen;
...
当你删除 return 语句时,GCC 会完全优化调用,因为它知道strlen
是没有副作用的函数,它实际上是一个保留名称,不应该被重载。我假设编译器可能会在这里给你一个警告,但唉,它没有,因为它不是必需的。
根据 C++14 [extern.names]/3,保留::strlen
:
使用外部链接声明的标准 C 库中的每个名称都保留给实现,以便在命名空间 std 和全局命名空间中用作具有外部"C"链接的名称。
以及使用保留名称 [reserved.names]/2 的效果:
如果程序在保留它的上下文中声明或定义名称,则此子句明确允许的名称除外,则其行为是未定义的。
因此,您的程序具有未定义的行为。
您在默认的 std 命名空间中定义了 strlen,从而覆盖了标准命名空间。
为什么有时调用 strlen,有时调用标准 strlen,可能与 strlen 的许多实现是宏而不是函数有关。它甚至可以在汇编程序中实现。
如果是宏,则将运行标准宏。此外,如果删除 return,优化器可以删除函数调用。您可以与 -O0 进行比较。
- 在提升multi_index容器中,是否定义了"default index"?
- #定义c-预处理器常量..我做错了什么
- 用C++中的一个变量定义一个常量
- 部分定义/别名模板模板参数
- C++映射:具有自定义类的运算符[]不起作用(总是返回0)
- #为""定义宏;静态";针对不同的上下文
- 如何确保C++函数在定义之前声明(如override关键字)
- 创建一个函数以在输入为负数或零时输出字符串.第一次执行用户定义的函数
- 当类在C++中定义时,有什么方法可以"register"类吗?
- 在命名空间中定义函数还是限定函数
- 此代码是否违反一个定义规则
- 编译C++时未定义的引用
- 不同翻译单元中不可重载的非内联函数定义
- 为什么在定义函数之前先声明它
- 有没有什么方法可以使用一个函数中定义的常量变量,也可以由c++中同一程序中的其他函数使用
- 在类定义之后定义一个私有方法
- 使用用户定义函数的字符串反转
- 用户定义函数中的指针和输入
- vscode g++链路故障:体系结构x86_64的未定义符号
- 手动定义的 strlens 的奇怪行为