0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

[C++] newの使い方

Posted at

質問:「C++言語におけるnewの使い方を教えてください.」
以下はChatGPTの回答.


C++におけるnew演算子は、動的メモリ割り当てを行うために使われます。これは、プログラムの実行中に必要なメモリをヒープ領域から確保するために使用されます。newを使うことで、動的にメモリを割り当て、必要なときにオブジェクトを作成することができます。

1. 基本的なnewの使い方

new演算子を使用すると、ヒープ領域にメモリを確保できます。newは、メモリの確保に成功するとポインタを返し、失敗するとstd::bad_alloc例外をスローします(デフォルトでは)。

使い方の例

#include <iostream>

int main() {
    // 整数型のメモリを動的に確保
    int* ptr = new int;

    // 確保したメモリに値を代入
    *ptr = 42;

    // メモリの値を出力
    std::cout << "The value is: " << *ptr << std::endl;

    // メモリの解放
    delete ptr;

    return 0;
}

解説

  • new int は、int型のメモリをヒープ領域に確保します。
  • ptrint*型のポインタで、newで確保されたメモリのアドレスを保持します。
  • delete演算子を使用して、確保したメモリを解放する必要があります。解放しないとメモリリークが発生します。

2. 配列の動的メモリ割り当て

newを使って配列のメモリも動的に確保することができます。配列の場合、new[]という構文を使います。

使い方の例

#include <iostream>

int main() {
    // 整数型の配列を動的に確保
    int* arr = new int[5];

    // 配列に値を代入
    for (int i = 0; i < 5; ++i) {
        arr[i] = i * 2;
    }

    // 配列の値を出力
    for (int i = 0; i < 5; ++i) {
        std::cout << "arr[" << i << "] = " << arr[i] << std::endl;
    }

    // メモリの解放
    delete[] arr;

    return 0;
}

解説

  • new int[5]は、int型の5つの要素を持つ配列をヒープ領域に確保します。
  • delete[]を使って、確保した配列のメモリを解放します。配列の場合は、delete[]を使うことを忘れないようにします。

3. クラスのインスタンスの動的割り当て

new演算子はクラスのインスタンスを動的に生成する場合にも使用されます。これにより、ヒープ領域にオブジェクトを生成することができます。

使い方の例

#include <iostream>

class MyClass {
public:
    MyClass() {
        std::cout << "MyClass constructor called!" << std::endl;
    }

    ~MyClass() {
        std::cout << "MyClass destructor called!" << std::endl;
    }

    void say_hello() {
        std::cout << "Hello from MyClass!" << std::endl;
    }
};

int main() {
    // クラスのインスタンスを動的に生成
    MyClass* obj = new MyClass();

    // メソッドを呼び出す
    obj->say_hello();

    // メモリの解放
    delete obj;

    return 0;
}

解説

  • new MyClass()MyClass型のインスタンスを動的に生成し、そのポインタをobjに格納します。
  • deleteを使ってオブジェクトのメモリを解放します。これにより、デストラクタが呼び出され、オブジェクトのクリーンアップが行われます。

4. メモリ割り当て失敗時のエラーハンドリング

new演算子は、メモリの割り当てに失敗するとstd::bad_alloc例外を投げます。これをキャッチして、エラーハンドリングを行うことができます。

使い方の例

#include <iostream>
#include <new>  // std::bad_allocを使うために必要

int main() {
    try {
        // メモリの割り当てを試みる
        int* ptr = new int[1000000000000];  // メモリが足りない場合
    }
    catch (const std::bad_alloc& e) {
        std::cerr << "Memory allocation failed: " << e.what() << std::endl;
    }

    return 0;
}

解説

  • std::bad_allocは、new演算子でメモリ割り当てが失敗したときにスローされる例外です。
  • これをtry-catchブロックでキャッチして、エラーメッセージを表示することができます。

5. newの代わりにスマートポインタを使用する

C++11以降では、手動でnewdeleteを使用するのではなく、スマートポインタ(std::unique_ptrstd::shared_ptr)を使ってメモリ管理を行うことが推奨されています。これにより、メモリリークやダングリングポインタの問題を防ぐことができます。

使い方の例(std::unique_ptr

#include <iostream>
#include <memory>

class MyClass {
public:
    MyClass() {
        std::cout << "MyClass constructor!" << std::endl;
    }

    ~MyClass() {
        std::cout << "MyClass destructor!" << std::endl;
    }

    void say_hello() {
        std::cout << "Hello from MyClass!" << std::endl;
    }
};

int main() {
    // unique_ptrでオブジェクトを管理
    std::unique_ptr<MyClass> ptr = std::make_unique<MyClass>();

    // メソッドを呼び出す
    ptr->say_hello();

    // 明示的にdeleteは不要。unique_ptrがスコープを抜けると自動でメモリが解放される

    return 0;
}

解説

  • std::make_unique<MyClass>()は、MyClass型のインスタンスを動的に確保し、それをstd::unique_ptrで管理します。
  • スマートポインタは、スコープを抜けると自動的にメモリを解放してくれるので、deleteを手動で呼び出す必要はありません。

結論

  • new演算子は、動的にメモリを割り当てるために使用します。
  • newで確保したメモリは、必ずdelete(または配列の場合はdelete[])で解放する必要があります。
  • C++11以降では、手動メモリ管理を避け、std::unique_ptrstd::shared_ptrなどのスマートポインタを使うことが推奨されます。
0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?