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

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

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

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

Q&A

解決済

1回答

5258閲覧

c++ クラス型ポインタのコピー

nasymt

総合スコア39

C++

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

0グッド

0クリップ

投稿2019/02/24 07:05

前提として、基底クラスBaseと、派生クラスA、Bがあります。
BaseUnitクラスでは、派生クラスA、Bのポインタが格納されたvectorがあります。
このBaseUnitクラスのvectorを、main関数の同じ型で別のvectorに実体をコピーしたいのですが、やり方がわからずうまくいきません。

以下のコードの出力結果は,
A val -> 1
A val -> 1
B val -> 2

になっているのですが、実体を別々に持って以下のような出力結果になるようにしたいです。
A val -> 0
A val -> 1
B val -> 2

初歩的な質問で申し訳ありませんが、よろしくお願いいたします。

c++

1class Base { 2 public: 3 virtual void print(){}; 4 virtual ~Base() = default; 5 6 int val; 7}; 8 9class A : public Base { 10 public: 11 void print() { cout << "A val -> " << val << endl; }; 12}; 13 14class B : public Base { 15 public: 16 void print() { cout << "B val -> " << val << endl; }; 17}; 18 19class BaseUnit { 20 public: 21 BaseUnit() { 22 units.push_back(new A()); 23 units.push_back(new B()); 24 } 25 26 vector<Base *> units; 27}; 28 29int main(){ 30 BaseUnit baseUnit; 31 vector<Base *> newUnits; 32 33 for(int i=0;i<3;i++){ 34 newUnits.push_back(new Base()); 35 if(i < 2){ 36 newUnits[i] = baseUnit.units[0]; 37 }else{ 38 newUnits[i] = baseUnit.units[1]; 39 } 40 newUnits[i]->val = i; 41 } 42 43 for(int i=0;i<newUnits.size();i++){ 44 newUnits[i]->print(); 45 } 46} 47

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

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

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

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

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

guest

回答1

0

ベストアンサー

こんにちは。

異なる実体が必要で、かつ、元のデータを引き継がせたいのであれば、実体を別途獲得してコピーする必要があります。

C++

1#include <iostream> 2#include <vector> 3using namespace std; 4 5class Base { 6 public: 7 virtual void print(){}; 8 virtual ~Base() = default; 9 virtual Base* clone() { cerr << "do not call Base::clone()\n"; abort(); } 10 11 int val; 12}; 13 14class A : public Base { 15 public: 16 void print() { cout << "A val -> " << val << endl; }; 17 Base* clone() { return new A(*this); } 18}; 19 20class B : public Base { 21 public: 22 void print() { cout << "B val -> " << val << endl; }; 23 Base* clone() { return new B(*this); } 24}; 25 26class BaseUnit { 27 public: 28 BaseUnit() { 29 units.push_back(new A()); 30 units.push_back(new B()); 31 } 32 33 vector<Base *> units; 34}; 35 36int main(){ 37 BaseUnit baseUnit; 38 vector<Base *> newUnits; 39 40 for(int i=0;i<3;i++){ 41 newUnits.push_back(new Base()); 42 if(i < 2){ 43 newUnits[i] = baseUnit.units[0]->clone(); 44 }else{ 45 newUnits[i] = baseUnit.units[1]->clone(); 46 } 47 newUnits[i]->val = i; 48 } 49 50 for(int i=0;i<newUnits.size();i++){ 51 newUnits[i]->print(); 52 } 53}

new A(*this);とnew B(*this);`はコンパイラが自動生成するコピー・コンストラクタを呼び出しています。

Base::clone()を安全に実装するのは手間なので異常終了させてます。(安易にBaseのコピー・コンストラクタでnewするとスライシングが発生する可能性が少しあるので)

なお、元からあったリークを修正していません。newUnits.push_back(new Base());にて獲得したBaseはdeleteされていませんのでリークします。

投稿2019/02/24 07:58

Chironian

総合スコア23272

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

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

nasymt

2019/02/24 08:09

早速のご回答ありがとうございます!無事思い通りの挙動を確認できました。 コピーコンストラクタはちゃんと理解していなかったので、大変勉強になりました。 リークについても、アドバイスありがとうございます。 確かにdeleteし忘れていたので、こちらも修正させていただきます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問