最安全合并原生C++数组的方式是先转为std::vector再操作;裸数组硬拼易越界或泄漏,std::array不支持运行时拼接,std::span仅提供视图不真正合并。

用 std::vector 合并数组最安全
原生 C++ 数组(int arr[5])大小固定、无内置拼接能力,硬凑容易越界或内存泄漏。实际开发中,几乎没人直接合并裸数组——先转成 std::vector 再操作才是合理路径。
常见错误现象:memcpy 手动拷贝时算错字节数、目标缓冲区没预留足够空间、忘记初始化导致读到垃圾值。
- 场景:两个
int数组长度已知,需合并为一个有序/无序结果 - 建议:用
std::vector构造后调用insert()或insert_range()(C++23) - 示例:
std::vector<int> a = {1, 2};<br>std::vector<int> b = {3, 4, 5};<br>a.insert(a.end(), b.begin(), b.end()); // a 变成 {1,2,3,4,5} - C++11 起支持,无需额外依赖;若必须返回裸数组,最后用
&a[0]取地址(确保非空)
手动合并裸数组必须自己管内存
如果真被限制只能用 int* 和 new,那就得自己分配新空间、逐个拷贝、再释放旧内存——这不是“拼接”,是重新构造。
容易踩的坑:new int[len1 + len2] 分配后没初始化,或漏掉某一段拷贝,或忘了 delete[] 原指针。
立即学习“C++免费学习笔记(深入)”;
- 参数差异:
memcpy比for循环快,但要求源和目标内存不重叠;重叠用memmove - 性能影响:频繁
new/delete触发堆分配开销,尤其小数组时比vector慢得多 - 示例:
int* merged = new int[len1 + len2];<br>memcpy(merged, arr1, len1 * sizeof(int));<br>memcpy(merged + len1, arr2, len2 * sizeof(int));
std::array 不支持运行时拼接
std::array<int, 5> 是编译期定长容器,size() 是 constexpr,无法在运行时改变容量。试图“合并”两个 std::array 实际上只能生成第三个不同尺寸的类型,而 C++ 不允许模板参数动态推导。
错误现象:auto c = a + b; 编译失败;c.insert(...) 不存在;std::copy 到另一个 std::array 需提前知道总长度。
- 使用场景:仅当两个数组长度完全确定且合起来仍为常量(如
std::array<int, 3>+std::array<int, 4>→std::array<int, 7>)才可行 - 建议:改用
std::vector,或者用std::tuple/ 结构体打包多个std::array,别强行拼 - 兼容性:C++11 起支持
std::array,但所有拼接逻辑都得手写,没通用接口
用 std::span 临时视图不等于合并
std::span(C++20)只是对已有内存的只读视图,它不分配、不拷贝、不拥有数据。所谓“合并两个 span”,只是把它们当成连续段去遍历,底层内存仍是分离的。
容易忽略的点:一旦原始数组生命周期结束,span 就变成悬垂指针;也不能拿它去传给需要真正连续内存的函数(比如某些 C 接口)。
- 适用场景:只读遍历、算法适配(如传给
std::sort),不想拷贝数据时的轻量替代 - 示例:
std::array<int, 2> a = {1, 2};<br>std::array<int, 3> b = {3, 4, 5};<br>std::span<const int> s1{a};<br>std::span<const int> s2{b};<br>// 没有 s1 + s2,只能分别处理 - 性能:零开销抽象,但功能边界非常清晰——它不是容器,也不解决合并问题
合并的本质是「新容器 + 数据迁移」,不是语法糖能绕开的。裸数组拼接代码看着短,出问题时调试成本远高于多写两行 vector。真要极致控制内存,也得从分配策略入手,而不是卡在“怎么连两个栈数组”这种伪需求上。










