gpt4 book ai didi

c++ - 为什么 msvc 编译器在显式调用析构函数时会发出未使用的变量

转载 作者:行者123 更新时间:2023-11-30 04:52:30 25 4
gpt4 key购买 nike

使用以下代码:

struct Foo {};

template<class T>
void Destruct(T *obj)
{
obj->~T();
}

int main(int /*argc*/, const char * /*argv*/[])
{
char buffer[sizeof(Foo)];
Destruct((Foo*)buffer);
return 0;
}

Visual Studio 2015 将针对未引用的参数发出警告:

warning C4100: 'obj': unreferenced formal parameter

这是合理的警告还是编译器中的错误?

在线重现:https://godbolt.org/z/xq96GU

编辑:将示例更新为完整示例

编辑 2:您需要启用/W4 才能在 visual studio 2015 中进行此操作,/W3 是不够的;还确认这不会发生在 2017 年。

编辑 3:对于 CNR,这里是命令行的输出,其中包含用于重现的所有参数:

>"C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\bin\CL.exe" /W4 test.cpp
Microsoft (R) C/C++ Optimizing Compiler Version 19.00.24215.1 for x86
Copyright (C) Microsoft Corporation. All rights reserved.

test.cpp
test.cpp(4): warning C4100: 'obj': unreferenced formal parameter
test.cpp(12): note: see reference to function template instantiation 'void Destruct<Foo>(T *)' being compiled
with
[
T=Foo
]
Microsoft (R) Incremental Linker Version 14.00.24215.1
Copyright (C) Microsoft Corporation. All rights reserved.

/out:test.exe
test.obj

编辑 4 在 godbolt.org 上添加了示例复制编辑5其实重现/W4就够了,不需要/Wall

最佳答案

恕我直言,这只是一种解释。

在专门函数中void Destruct<Foo>(Foo *obj) ,传递的对象仅用于调用一个空的析构函数。不涉及虚拟,没有变量变化值,没有发生IO。长话短说,该调用不会直接或间接观察到任何结果,因此编译器可以对其进行优化。因此,当编译器说传递的对象在该特定特化中未使用时,它是正确的。

此外,每个标准都需要很少的诊断,但据我所知,没有什么能阻止编译器对可疑代码发出警告。这正是这里发生的事情:编译器警告您,对于您的代码,对析构函数的调用将是空操作,因此代码是可疑的。但是没有编译错误,而且我假设程序将毫无问题地运行,尽管转换违反了严格的别名规则。

所以我的意见是这个警告远非必需,不发出它的编译器是正确的,但由于代码奇怪,你不能责怪编译器善意地警告你说程序员,我希望你知道你在这里做什么,因为我无法理解其背后的基本原理......

关于c++ - 为什么 msvc 编译器在显式调用析构函数时会发出未使用的变量,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54346875/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com