不能使用std :: unique_ptr <t>,T是前向声明</t>

时间:2015-02-07 19:08:00

标签: c++ c++11 unique-ptr forward-declaration

现在首先,我知道unique_ptr&lt;&gt;的一般问题和转发声明,如Forward declaration with unique_ptr?

考虑这三个文件:

A.H

#include <memory>
#include <vector>

class B;

class A
{
public:
    ~A();

private:
    std::unique_ptr<B> m_tilesets;
};

C.cpp

#include "A.h"

class B {

};

A::~A() {

}

的main.cpp

#include <memory>

#include "A.h"

int main() {
    std::unique_ptr<A> m_result(new A());
}

发出g++ -std=c++11 main.cpp C.cpp会产生以下错误:

In file included from /usr/include/c++/4.8/memory:81:0,
                 from main.cpp:1:
/usr/include/c++/4.8/bits/unique_ptr.h: In instantiation of ‘void std::default_delete<_Tp>::operator()(_Tp*) const [with _Tp = B]’:
/usr/include/c++/4.8/bits/unique_ptr.h:184:16:   required from ‘std::unique_ptr<_Tp, _Dp>::~unique_ptr() [with _Tp = B; _Dp = std::default_delete<B>]’
A.h:6:7:   required from here
/usr/include/c++/4.8/bits/unique_ptr.h:65:22: error: invalid application of ‘sizeof’ to incomplete type ‘B’
  static_assert(sizeof(_Tp)>0,

这是真的,B在A.h的第6行中是不完整的类型 - 但这不是A的析构函数所在! g ++似乎为A生成析构函数,即使我提供了析构函数。 Ac的析构函数位于C.cpp第7行,其中B是完美定义的类型。为什么我会收到此错误?

2 个答案:

答案 0 :(得分:15)

您还需要将A&#39的构造函数放在C.cpp中:

A.H

#include <memory>
#include <vector>

class B;

class A {
public:
     A();
    ~A();

private:
    std::unique_ptr<B> m_tilesets;
};

C.cpp

#include "A.h"

class B {

};

A::~A() {

}

A::A() {

}

this answer。构造函数也需要访问完整类型。如果在构造期间抛出异常,它可以调用删除器。

答案 1 :(得分:6)

隐式定义的特殊成员函数是内联的,导致类型不完整的问题。正如link中的Chris's answer所示,所有非委托构造函数都可能会调用析构函数。这包括副本(在本例中删除)和移动构造函数。因此,当您处理涉及不完整类型的非静态成员时,请明确默认源文件中的定义,从而确保它们不是内联定义的。

在标题中:

A();
~A();
A(const A&);
A(A&&);

来源:

A::A() = default;
A::~A() = default;
A::A(const A&) = default;
A::A(A&&) = default;
相关问题