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

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

ただいまの
回答率

87.37%

コピーコンストラクタの仕組みを教えてください

解決済

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 644

score 9

以下のプログラムは参考書に書かれてあるプログラムです。

(あ)の、コピーコンストラクタの仕組みがイマイチ分かりません。
メモリの二重解放を防ぐために行われているという事は分かるのですが、ここの p はどのオブジェクトのpになるんでしょうか。
main()内でgetval()の仮引数に指定されているオブジェクトは、o になりますよね?
p が何をしているのかが分かりません。
どなたか宜しくお願いします。

// このプログラムにはエラーがある

include <iostream>

include <cstdlib>

using namespace std;

class myclass {
int *p;
public:
myclass(int i);
myclass(const myclass &ob);
~myclass() { delete p; }
friend int getval(myclass o);
};

myclass::myclass(int i)
{
p = new int;

if(!p) {
cout << "メモリ割り当てエラー\n";
exit(1);
}

*p = i;

}

myclass::myclass(const myclass &ob) ・・・(あ)
{
p = new(nothrow) int;

if(!p){
cout <<"メモリ割り当てエラー\n";
exit(1);
}

*p = *ob.p;

}

int getval(myclass o)

return *o.p; // 値を取得する
}

int main()
{
myclass a(1), b(2);

cout << getval(a) << " " << getval(b);
cout << "\n";
cout << getval(a) << " " << getval(b);
cout << "\n";

return 0;
}

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

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • hoshi-takanori

    2020/06/01 05:27 編集

    本来は *a.p と *b.p が別の値を持てるようにするためですが、メモリの二重解放を防ぐ意味もあります。
    よく見たら getval の引数はコピー渡しされているので、コピーコンストラクタがないとメモリの二重解放になりますね。

    キャンセル

  • iwanami

    2020/06/01 14:06

    僕が参考書で学んだのはメモリの二重解放だけだったので、*a.pと*b.pが別の値を持てるようにするため、というのは知りませんでした。
    ただ今回はメモリの二重解放に焦点を当てていて、(あ)のコピーコンストラクタの定義でなぜ二重解放が防げているのかが分かりません。

    キャンセル

  • iwanami

    2020/06/01 16:10

    回答してくれた皆さんありがとうございました。

    キャンセル

回答 1

checkベストアンサー

+4

二重解放とならぬよう...

myclass::myclass(const myclass &ob) { // ・・・(あ)
  p = new(nothrow) int; // 新たな領域を確保し 
  if(!p){
    cout <<"メモリ割り当てエラー\n";
    exit(1);
  } 
  *p = *ob.p; // ナカミを複製する 
} 

なので
myclass a(123);
myclass b = a;
としたとき、a.p と b.p は別領域となります(からa,bが廃棄されるとき二重解放になりません)

[追記] 領域のnew/delete の様子をプリントしてみました:

#include <iostream>
#include <cstdlib>
using namespace std;

class myclass {
    int* p;
public:
    myclass(int i);
    myclass(const myclass& ob);
    ~myclass() { std::cout << "delete " << (void*)p << "\n"; delete p; }
    friend int getval(myclass o);
};
myclass::myclass(int i)
{
    p = new int;
    std::cout << "new " << (void*)p << "\n";
    if (!p) {
        cout << "メモリ割り当てエラー\n";
        exit(1);
    }
    *p = i;
}
myclass::myclass(const myclass& ob)
{
    p = new(nothrow) int;
    if (!p) {
        cout << "メモリ割り当てエラー\n";
        exit(1);
    }
    std::cout << "new " << (void*)p << "\n";
    *p = *ob.p;
}
int getval(myclass o)
{
std::cout << "getval\n";
    return *o.p; // 値を取得する
}
int main()
{
    myclass a(1), b(2);
    cout << getval(a) << " " << getval(b);
    cout << "\n";
    cout << getval(a) << " " << getval(b);
    cout << "\n";
    return 0;
}


実行結果:

new 000002324AA76DE0
new 000002324AA76D40
new 000002324AA76E80
getval
delete 000002324AA76E80
1 new 000002324AA76E10
getval
delete 000002324AA76E10
2
new 000002324AA76DD0
getval
delete 000002324AA76DD0
1 new 000002324AA76CF0
getval
delete 000002324AA76CF0
2
delete 000002324AA76D40
delete 000002324AA76DE0

getvalの前後でnew/deleteされてて、そのポインタ値は組になってます。
つまり、getvalの直前でコピーを作り、直後でそのコピーを廃棄しています。
なので二重解放になりません。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/06/01 15:32

    度々すいません、回答に気づかなかったので上のコメントは15:04までの返信です。今から理解したいと思います。

    キャンセル

  • 2020/06/01 16:02 編集

    (一つ上の回答も15:27までの返信です。返信遅くてすいません。)

    なるほどー!!
    今漸く分かりました。

    理解出来ない事が多くて少し逆切れ気味だった所もあり(逆切れしているつもりはないんですが、そう見えるコメントは合ったかと思います)、僕の理解が足りないために長く付き合わせてしまってすいません。
    epistemeさんのプログラムのおかげで流れを目で追って理解する事ができました。
    ありがとうございます。

    キャンセル

  • 2020/06/01 16:10 編集

    まだまだ自分が初心者だと分かりました。
    episteme さんには分かりやすいプログラムまで記述してくださり感謝の気持ちで一杯です。

    回答してくれた皆さんありがとうございました。

    キャンセル

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

  • ただいまの回答率 87.37%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る