1. 導入
C++において、動的メモリ管理は避けて通れない重要なタスクです。new演算子で確保したメモリは、不要になった時点で適切に解放しなければ「メモリリーク」を引き起こし、システムの安定性を損ないます。本記事では、delete演算子の正しい使い方と、実務で陥りやすいトラブルを避けるための基本作法について解説します。
2. 基礎知識
C++のメモリ管理において、ヒープ領域(フリーストア)にオブジェクトを配置するのがnew演算子であり、その領域を解放し、対象オブジェクトのデストラクタを呼び出すのがdelete演算子です。
重要なポイントは、「確保した方法と解放の方法を一致させる」ことです。単一のオブジェクトを確保した場合はdeleteを使い、配列を確保した場合はdelete[]を使わなければなりません。この不一致は未定義動作を引き起こす原因となります。
3. 実装/解決策
delete演算子を使用する際は、以下の手順を徹底することが重要です。
1. ポインタの有効性を確認する:deleteはnullptrに対して実行しても安全(何もしない)ですが、無効なアドレスに対して実行するとクラッシュします。
2. 解放後のポインタを無効化する:deleteした後のポインタは「ダングリングポインタ(ゴミを指すポインタ)」となります。これに再度アクセスしないよう、delete直後にnullptrを代入するのが実務での定石です。
4. サンプルプログラム
以下のコードは、単一オブジェクトおよび配列の解放の作法を示したものです。
include
class Resource {
public:
Resource() { std::cout << "リソース確保\n"; }
~Resource() { std::cout << "リソース解放(デストラクタ)\n"; }
};
int main() {
// 1. 単一オブジェクトの生成と解放
Resource res = new Resource();
// 処理...
delete res; // デストラクタが呼び出され、メモリが解放される
res = nullptr; // 二重解放や誤アクセスを防ぐため、安全のためにnullptrを代入
// 2. 配列の生成と解放
Resource resArray = new Resource[3];
// 配列の場合は delete[] を使用する(必須)
delete[] resArray;
resArray = nullptr;
return 0;
}
5. 応用・注意点
現場レベルでの開発において、手動でのdelete管理はバグの温床になりがちです。
スマートポインタの活用
現代のC++(C++11以降)では、std::unique_ptrやstd::shared_ptrを使用することが強く推奨されます。これらを使用すれば、スコープを抜ける際に自動的にdeleteが呼び出されるため、メモリリークのリスクを大幅に低減できます。
注意点:
・deleteした後のポインタに対して、再度deleteを呼び出す「二重解放(Double Free)」は致命的なエラーです。
・継承関係にあるクラスを削除する場合、基底クラスのデストラクタが仮想関数(virtual)であることを確認してください。そうでない場合、派生クラスのデストラクタが呼び出されず、メモリリークや動作不良の原因となります。
基本的にはスマートポインタへの移行を検討しつつ、レガシーコードの保守などでdeleteを使う際は、上記の手順を厳守してください。

コメント