推荐用new分配连续内存的二维数组:先一次性分配rowscols个int的连续内存,再分配rows个int指针数组,通过指针偏移(data+i*cols)使各arr[i]指向对应行首,确保内存连续、缓存友好且避免内存泄漏。

用 new 分配连续内存的二维数组(推荐)
二维数组本质是“一维数组的数组”,但直接 new int*[rows] 再循环 new int[cols] 会产生多段不连续内存,增加缓存失效风险,也容易漏删导致内存泄漏。
更稳妥的做法是只分配一块连续内存,再用指针偏移模拟二维访问:
int** create2DArray(int rows, int cols) { int* data = new int[rows * cols]; // 1 次分配 int** arr = new int*[rows]; for (int i = 0; i < rows; ++i) { arr[i] = data + i * cols; // 每行首地址 = data + 行偏移 } return arr; }
使用时仍可写 arr[i][j],但底层是连续布局。释放需两步:delete[] arr[0](即 data),再 delete[] arr。
用 vector> 初始化二维数组(适合快速开发)
如果不需要裸指针或极致性能,std::vector 是最安全的选择。它自动管理内存,支持初始化大小和默认值:
立即学习“C++免费学习笔记(深入)”;
std::vector> mat(rows, std::vector(cols, 0));
这会创建 rows 行、每行 cols 个 0 的二维结构。注意:内部每个 vector 是独立分配的,内存不连续;若后续要传给 C 风格接口(如 OpenGL 或 opencv 的 cv::Mat),不能直接用 &mat[0][0] 取首地址——可能崩溃或读错数据。
常见误用:mat.resize(rows, std::vector —— 这会复用原 vector 的 capacity,但新行仍是空的,需额外赋值。
静态/栈上二维数组的初始化限制
c++ 不允许用变量声明栈上二维数组(如 int arr[n][m]),这是 VLA(变长数组),属于 C99 特性,C++ 标准不支持(GCC 允许但属扩展,不可移植)。
若行列已知为编译时常量,可用:
constexpr int R = 3, C = 4; int arr[R][C] = {}; // 全零初始化
或用 std::array 替代:
std::array, R> arr = {}; // 类型安全,支持范围 for
注意:std::array 大小必须是常量表达式,不能用运行时变量。
释放动态二维数组时最容易漏掉的点
用 new[] 分配的内存,必须严格匹配 delete[];混用 delete 会导致未定义行为。对于连续内存方案,有两个指针要删:
-
delete[] arr[0]→ 释放数据块(data) -
delete[] arr→ 释放行指针数组
顺序不能反:如果先删 arr,就丢失了 arr[0] 的地址,无法释放数据块。封装成类时,务必在析构函数中按此顺序清理;用 RaiI(如智能指针)可避免手写释放逻辑,但需注意 std::unique_ptr 和 std::unique_ptr 的类型写法差异。
实际项目里,除非有明确性能或接口约束,优先用 vector 或封装好的矩阵类(如 Eigen),裸 new/delete 容易出错且难维护。