C++中的delete和delete[]有什么区别_C++内存释放与delete使用解析

delete用于释放单个对象,delete[]用于释放对象数组,必须与new和new[]匹配使用;对于类类型,错误混用会导致析构函数未被正确调用,引发未定义行为。

在C++中,deletedelete[] 都用于释放动态分配的内存,但它们的使用场景和底层行为有重要区别。错误地混用可能导致未定义行为,甚至程序崩溃。

1. delete 与 delete[] 的基本用途

delete 用于释放通过 new 分配的单个对象:

int* p = new int(10);
delete p;  // 正确:释放单个对象

delete[] 用于释放通过 new[] 分配的对象数组:

int* arr = new int[10];
delete[] arr;  // 正确:释放对象数组

关键点是:用哪个操作符分配,就用对应的释放操作符。匹配使用才能确保正确调用析构函数并释放内存。

2. 析构函数调用差异

对于内置类型(如 int、double),误用 delete 和 delete[] 可能不会立刻出错,但对于类类型,问题会更严重。

例如:

class MyClass {
public:
    ~MyClass() { cout << "析构函数被调用\n"; }
};

MyClass* obj = new MyClass(); delete obj; // 正确调用一次析构函数

MyClass* arr = new MyClass[5]; delete[] arr; // 正确调用5次析构函数,每个元素一次

如果对数组使用 delete 而不是 delete[],通常只会调用第一个元素的析构函数,其余对象的析构函数不会被调用,造成资源泄漏。

3. 内存管理机制的不同

编译器在使用 new[] 分配数组时,可能会在内存块前额外存储数组长度或其他元信息,以便 delete[] 能正确遍历并析构每一个元素。

delete 不会读取这些信息,它只处理单个对象。因此:

  • delete 释放 new[] 的数组 → 未定义行为,可能漏掉析构、内存损坏
  • delete[] 释放 new 的单个对象 → 未定义行为,可能读取非法元数据

4. 实际建议与最佳实践

为避免出错,推荐以下做法:

  • 始终成对使用:new 对应 delete,new[] 对应 delete[]
  • 优先使用智能指针(如 std::unique_ptr, std::shared_ptr)和容器(如 std::vector)
  • 对于数组,使用 std::vector 或 std::array 替代原始数组
  • 如果必须用 new[],务必用 delete[] 释放

例如:

std::unique_ptr arr = std::make_unique(10);
// 自动调用 delete[],无需手动管理

基本上就这些。只要记住分配方式决定释放方式,再尽量用现代C++替代裸指针,就能避免大部分问题。