質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.50%
C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

1回答

8116閲覧

【C++】no viable conversionのエラーが解決できない

Yuya4

総合スコア25

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

0クリップ

投稿2019/06/11 16:33

編集2019/06/12 02:07

前提・実現したいこと

C++初心者です。
子クラスのインスタンスを親クラス型の変数に代入しようとするとエラーが発生し、解決方法が分かりません。

発生している問題・エラーメッセージ

error: no viable conversion from 'shared_ptr<ChildClass>' to 'shared_ptr<ParentClass>'

普通、子クラスのインスタンスは親クラスの変数に代入できるはずですが、なぜかこうなりました。
このような状況になりうる原因があれば、教えていただきたいです。

該当のソースコード

以下のようなソースコードを作成しました。
Apple LLVM version 8.1.0 (clang-802.0.42)で動作させてエラーが発生します。

parent_class.h

cpp

1using std::shared_ptr; 2class ParentClass:public std::enable_shared_from_this<ParentClass> 3{ 4public: 5 ~ParentClass(); 6 void method1(); 7}

parent_class.cpp

cpp

1#include "parent_class.h" 2ParentClass::~ParentClass() 3{ 4} 5void ParentClass::method1(){ 6 // 何らかの処理 7}

child_class.h

cpp

1#include "parent_class.h" 2using std::shared_ptr; 3class ChildClass:public ParentClass, public std::enable_shared_from_this<ChildClass> 4{ 5public: 6 ~ChildClass(); 7 void method2(); 8}

child_class.cpp

cpp

1#include "child_class.h" 2ChildClass::~ChildClass() 3{ 4} 5void ChildClass::method2() 6{ 7 // 何らかの処理 8}

main.cpp

cpp

1#include "parent_class.h" 2#include "child_class.h" 3int main(int argc, char *argv[]) 4{ 5 shared_ptr<ParentClass> pcPtr = std::make_shared<ChildClass>(); // エラー発生箇所 6}

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

yohhoy

2019/06/12 01:27 編集

ご利用中のC++コンパイラ名(gcc/clang/VisualC++など)と、正確なバージョンはわかりますか?
Yuya4

2019/06/12 02:14 編集

Apple LLVM version 8.1.0 (clang-802.0.42)を使っています。 enable_shared_from_this<T>からの継承は意図的というか、既存のソースコードがそうなっていたのでそのまま使っているという感じです。 すみません!
guest

回答1

0

ベストアンサー

こんにちは。

yohhoyさんのコメントを見て、全体的に間違っていたので修正します。すいません。

元の回答の前半は正しいのですが、shared_ptr<T>にはshard_ptr<Y>の参照を受け取るコンストラクタが定義されていて、TをYへ変換できる(TがYの既定クラスの時も該当)なら有効に機能します。
従って、ご提示されているようなエラーは出ない筈です。

面倒なのでファイルを1つにまとめましたが、確かにエラーはでませんでした。
wandbox

お使いのコンパイラが古い等の可能性はないでしょうか?


【元の回答】

子クラスのインスタンスは親クラスの変数に代入できる

その通りです。ただ、私も昔はまったことがあるのですが、shared_ptr<ChildClass>は、shared_ptr<ParentClass>の派生クラスではありません。

もしも以下のような記述ができればそうなりますが、そもそもこのような定義はできません。

class shard_ptr<ChildClass> : public shared_ptr<ParentClass> { };

このケースでは、make_sharedは事実上使えません。(make_shared<T>はTの領域を確保しますが、そこにTの領域にTの派生クラスは通常は入りませんので。)

コンパイルしていませんが、下記なら行けると思います。

C++

1shared_ptr<ParentClass> pcPtr(new ChildClass);

投稿2019/06/11 18:59

編集2019/06/12 02:55
Chironian

総合スコア23272

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

Chironian

2019/06/12 02:49

majiponiさん ああ、その通りですね。Yuya4さん、デストラクタを仮想関数にしておかないと基底クラスへのポインタを delete すると多くの場合問題が起きます。(派生クラスのデストラクタが呼ばれないため) yohhoyさん あらら、本当ですね。回答を訂正します! なるほど、以前ハマったのは自分で定義したクラスだったからのようです。 別クラスからのコンストラクタを明示的に定義すればよかったのか~。
yohhoy

2019/06/13 23:00 編集

majiponi さん: std::shared_ptrを適切に使うのであれば、基底クラスデストラクタは非仮想関数でも大丈夫ですよ。 先の例示コード https://wandbox.org/permlink/fF0aw4RHZrnClZG4 でも意図的にvirtualを外していますが、正しく ~ChildClass → ~ParentClass が呼ばれています。 Chironian さんコードのように make_shared経由しなくとも https://wandbox.org/permlink/MgPRROvQ3Fq2rZDG 問題ありません。
Chironian

2019/06/12 09:02

yohhoyさん shared_ptrなかなか深いですね。しかし、どうやって実装しているのだろう?
yohhoy

2019/06/12 10:50 編集

std::shared_ptrクラステンプレート内部実装では Type Erasure と呼ばれるテクニックが(通常は)使われています。 簡単に言うと shared_ptr内部のオブジェクト保持クラスに真の型情報を持たせることで、外部インタフェース(shared_ptr<ParentClass>)と内部実装(value-holder-impl<ChildClass>)を分離します。内部実装ではオブジェクトの正しい型を知っているので、ChildClassデストラクタを適切に呼び出せるという仕組みですね。
Chironian

2019/06/12 12:47

yohhoyさん ありがとうございます。なるほど。Type Erasere 把握してます。 書いたこともあるけど、何故か苦手なテクニックです。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.50%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問