web-dev-qa-db-ja.com

std :: unique_ptr <T>を使用できません。Tは前方宣言です。

最初に、unique_ptr <>の一般的な問題と、 nique_ptrを使用した前方宣言? のような前方宣言を認識しています。

次の3つのファイルについて考えてみます。

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のデストラクタを生成しているようです。 AのデストラクタはC.cppの7行目にあり、Bは完​​全に定義された型です。なぜこのエラーが発生するのですか?

18
hllnll

また、Aのコンストラクターを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() {

}

この回答 を参照してください。コンストラクターは、完全な型にもアクセスする必要があります。これは、構築中に例外がスローされた場合に削除機能を呼び出すことができるようにするためです。

20
Chris Drew

暗黙的に定義された特殊メンバー関数はインラインであり、不完全な型の問題につながります。 link from Chris's answer が示すように、すべての非委任コンストラクターがデストラクタを呼び出す可能性があります。これには、copy(この場合は削除)およびmoveコンストラクターも含まれます。したがって、不完全な型を含む非静的メンバーを扱う場合は、ソースファイルの定義を明示的にデフォルト設定して、インラインで定義されないようにします。

ヘッダー内:

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

ソース:

A::A() = default;
A::~A() = default;
A::A(const A&) = default;
A::A(A&&) = default;
8
Pradhan