- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
这个例子展示了编译器(msvc14、gcc、clang)的奇怪行为,但我没有找到解释。
当我们实现 pipml idiom 并使用前向声明时,我们需要考虑 unique_ptr 具有不完整类型的特定行为。这个案例被提及here和 here .
但是当我们将转发类的定义移动到另一个头文件并在稍后使用客户端类时将头文件包含在一个地方时,编译器会变得疯狂 - 在析构函数声明的某些特殊情况下,他们会说不完整的类型。
这是一个最小的例子。如果取消注释“#define CASE_2”或“#define CASE_3”并尝试构建它,将会出现编译错误。
文件 foo.h
#ifndef FOO_H
#define FOO_H
class Foo{};
#endif // FOO_H
文件base.h
#ifndef BASE_H
#define BASE_H
#include <memory>
//#define CASE_1
//#define CASE_2
//#define CASE_3
class Foo;
class Base
{
public:
#if defined(CASE_1)
~Base() = default; // OK!
#elif defined(CASE_2)
~Base() {}; // error: invalid application of 'sizeof' to incomplete type 'Foo'
#elif defined(CASE_3)
~Base(); // error: invalid application of 'sizeof' to incomplete type 'Foo'
#endif
// OK!
private:
std::unique_ptr<Foo> m_foo;
};
#endif // BASE_H
文件base.cpp
#include "base.h"
#if defined(CASE_3)
Base::~Base()
{
}
#endif
文件main.cpp
#include "foo.h" // No matter order of this includes
#include "base.h" //
int main()
{
Base b;
}
最佳答案
我相信,这与 C++ 标准 12.4/6 有关。
A destructor that is defaulted and not defined as deleted is implicitly defined when it is odr-used (3.2) to destroy an object of its class type (3.7) or when it is explicitly defaulted after its first declaration.
当您的析构函数默认时,它只会在使用 ODR 时被定义,即当 Base
对象被销毁时。在你的代码片段中,没有这种类型的对象被销毁,因此,程序编译 - 因为 unique_ptr 的 deleter 实际上并没有在任何地方调用 - 它只被 Base
析构函数调用,它没有定义在这种情况下。
当您提供一个用户定义的析构函数时,它是就地定义的,并且程序变得格式错误,因为您无法析构不完整类型的 unique_ptr
对象。
顺便说一句,析构函数声明
,但不是定义
(如~base();
)不会产生编译错误同样的道理。
关于c++ - 不是默认析构函数导致不完整的类型错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45328143/
我开始考虑在我 future 的项目或重构中实现控制反转容器,我想知道在正确设计依赖项时哪些原则(除了 GoF 模式)可能需要牢记在心。假设我需要构建一个简单的控制台应用程序,如果它可以访问互联网,它
假设我有一个 RxC contingency table 。这意味着有 R 行和 C 列。我想要一个维度为 RC × (R + C − 2) 的矩阵 X,其中包含行的 R − 1 “主效应”以及列的
我正在尝试使用 DKMS 为正在运行的内核 (4.4) 构 build 备树覆盖。我天真的 Makefile 如下: PWD := $(shell pwd) dtbo-y += my-awsome-o
我有一个 sencha touch 项目。我是用 phonegap 2.9 构建的,并且可以正常工作 device.uuid 返回到设备 ID。当我尝试使用 3.1 device.uuid 构建时抛出
我在安装了 Xcode 4.5.1 的 Mt Lion 上运行。 默认情况下,当我构建并部署到 iOS 5.1 设备时,显示会在我旋转设备时旋转,但当我部署到 iOS 6 模拟器或运行 iOS 的 i
我正在尝试使用 Google Analytics Reporting API v4 构建多折线图。 一张图表,其中我按每天的 session 计数为每个设备(台式机/平板电脑/移动设备)设置了一条线。
我一生都无法使用 xcode 组织者“自动设备配置”中的“团队配置配置文件”在 xcode 4.0.1 中将我的应用程序构建到我的 iPad 上。 该应用程序完美地构建到模拟器,但当我构建到 iPad
我是一名优秀的程序员,十分优秀!