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

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

ただいまの
回答率

87.37%

クラス配列でデスストラクタが動いてしまってる理由が知りたい。

受付中

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 785
退会済みユーザー

退会済みユーザー

main関数部の一行の配列のコードなのですが派生クラスであるderiverdクラスを設定したときになぜコピーコンストラクタとデストラクタが動いてしますのでしょうか?コピーコンストラクタは宣言だからとかそれ以上の理由はないと判断したとしても
1,デストラクタが呼ばれる理由が知りたいです。

#include <iostream>
#include <ctime>
#include <cstdlib>

class base {
private:
    int *ar = NULL;
    int d = 0;

public:

    base(int x) {
        d = x;//配列の個数
        ar = new int[x];
        for (int i = 0; i < x; i++)
        {
            ar[i] = rand() % 100;
        }

        std::cout << "base\n";
    }

    base(const base& x)//コピーコンストラクタ
    {
        if (this != &x)
        {
            if (this->d == x.d)
            {
                for (int i = 0; i < d; i++)
                {
                    this->ar[i] = x.ar[i];
                }
            }
            else {
                if (this->d < x.d)
                {
                    //int r = x.d - this->d;
                    for (int i = 0; i < this->d; i++)
                    {
                        this->ar[i] = x.ar[i];
                        //delete(this->ar);

                    }
                }
                else {
                    //int r = this->d - x.d;
                    for (int i = 0; i < x.d; i++)
                    {
                        this->ar[i] = x.ar[i];
                        //        delete(x.ar);
                    }
                }
            }
        }
        std::cout << "copy constructor\n";
    }

    ~base()
    {
        delete(ar);
        std::cout << "base Destructor\n";
    }

    //void f()//オーバーライド
    virtual void f()//オーバーライド
    {
        std::cout << "base f()\n";
        //std::cout << a << std::endl;

    }

    void p()
    {
        for (int i = 0; i < d; i++)
        {
            std::cout << "ar[" << i<<"]: "<< ar[i] << std::endl;
        }
    }

    base& operator = (const base& x)
    {
        std::cout << "base operator  = \n";
        if (this != &x)
        {
            if (this->d == x.d)
            {
                for (int i = 0; i < d; i++)
                {
                    this->ar[i] = x.ar[i];
                }
            }
            else {
                if (this->d < x.d)
                {
                    //int r = x.d - this->d;
                    for (int i = 0; i < this->d; i++)
                    {
                        ar[i] = x.ar[i];
                        //delete(this->ar);

                    }
                }
                else {
                    //int r = this->d - x.d;
                    for (int i = 0; i < x.d; i++)
                    {
                        this->ar[i] = x.ar[i];
                //        delete(x.ar);
                    }
                }
            }

            return *this;
        }
    }
    void x(int t) {
        std::cout << t <<"x()\n";
    }


};

class derived : public base {
public:
    derived(int t):base(t)
    {
        std::cout << "deirved\n";
    }

    void x(int x) {
        std::cout << "d x()\n";
        std::cout << x;
    }

    ~derived()
    {
        std::cout << "derived Destructor\n";
    }
    void df() { std::cout << "df()"; }
    void f() override
    {
        std::cout << "deiverd f()\n";
        //std::cout << a << std::endl;
    //    return 3;

    }
};

int main()
{

    base* idx = new base[5]{ derived(1),base(1),base(1),base(1),base(1) };












    int c_ = getchar();
    return 0;
}
  • 気になる質問をクリップする

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 2

+2

base d[] = {derived(1)};

base d = derived(1);

と、大体等しく

コピーして新しくbaseのインスタンスを作る事になるから

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2019/08/11 00:04

    もう少し詳細にお願いですか、base b = derived(1);はまずコピーコンストラクタが実行されてから
    デストラクタが動いていますが

    キャンセル

  • 2019/08/11 00:30

    横からすみません。
    スコープに着目するとasmさんが書かれた内容を理解しやすいと思います。
    base d[] = { derived(1), };

    base d[] = { (0番目の要素) = derived(1) };
    とまずオブジェクトのコピーが行われます(ここでコピーコンストラクタが呼ばれる)。
    そのあと、処理は { } のスコープを外れるので、derived(1)で作ったインスタンスが破棄されます。
    そのためデストラクタが動くと考えてください。

    キャンセル

0

main関数内でインスタンスが使われていない>なら不要なものなので破棄しておく
ということだと。
べつにおかしなはなしではないですね

#で、ローカルスコープで生成したインスタンスはスコープから抜けるときに破棄されます

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

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

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

関連した質問

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