c++如何合并两个vector_c++ vector insert与merge操作【进阶】

2次阅读

vector::insert 不能实现归并排序意义上的合并,仅支持拼接;需用 std::merge 实现有序归并,且要求输入已排序、结果写入新容器或通过临时副本安全更新。

c++如何合并两个vector_c++ vector insert与merge操作【进阶】

vector::insert 能不能直接合并两个 vector?

能,但不是“合并”意义上的合并——insert 是在指定位置插入元素,不自动去重、不保证有序、不处理内存重分配逻辑。它只是把另一个 vector 的所有元素逐个拷贝进去。

常见误用:以为 vec1.insert(vec1.end(), vec2.begin(), vec2.end()) 会像 std::merge 那样归并两个已排序序列,其实不会。它只是拼接(concatenation),结果等价于 vec1.insert(vec1.end(), vec2) 或更简洁的 vec1.insert(vec1.end(), vec2.begin(), vec2.end())

  • 如果 vec1vec2 都已升序,但你想得到一个升序结果,insert 不满足需求,必须用 std::merge
  • insert 可能触发多次内存重分配(尤其未预留空间时),影响性能
  • 若需保留 vec2 原内容,insert 安全;若想“移动”而非拷贝,应考虑 std::move_iterator

std::merge 要求两个输入 range 必须已排序

std::merge 是归并算法,不是容器操作函数,它不修改原容器,只把归并结果写入目标迭代器。它要求:first1last1first2last2 都是升序(默认)或按同一自定义比较器排好序的范围。

典型用法:

立即学习C++免费学习笔记(深入)”;

std::vector a = {1, 3, 5, 7}; std::vector b = {2, 4, 6, 8}; std::vector result; result.reserve(a.size() + b.size()); // 预留空间避免多次 realloc std::merge(a.begin(), a.end(), b.begin(), b.end(), std::back_inserter(result));
  • 漏掉 reserveback_inserter 每次 push_back 可能触发 vector 扩容,O(n) 次内存拷贝
  • 传错迭代器范围(比如 b.end() - 1)→ 结果缺失或越界
  • 没确认两容器是否真有序 → 行为未定义,可能输出乱序甚至崩溃
  • 若需降序归并,必须传入 std::greater{} 作为第五个参数,且两个输入也得是降序

移动语义下高效合并:避免无谓拷贝

vec2 后续不再使用,可用移动迭代器把元素“搬”过去,减少深拷贝开销,尤其对含动态内存的对象(如 std::String、自定义类)效果明显。

两种常用方式:

  • insert + std::make_move_iteratorvec1.insert(vec1.end(), std::make_move_iterator(vec2.begin()), std::make_move_iterator(vec2.end()));
  • std::moveclear(更直观):vec1.insert(vec1.end(), std::make_move_iterator(vec2.begin()), std::make_move_iterator(vec2.end())); vec2.clear();

注意:std::move 本身不移动,只是转换为右值引用;真正触发移动的是容器的 insert 或 assign 实现。若 vec2 元素类型没有移动构造函数,仍会退化为拷贝。

merge 后想原地更新?别直接往自己里面 merge

不能对同一个 vector 执行 std::merge(v.begin(), v.end(), v.begin(), v.end(), v.begin()) —— 这是未定义行为:读写同一内存区域且有重叠,编译器不保证结果,运行时可能数据错乱。

正确做法只有两种:

  • 写入新 vector(推荐,语义清晰、安全)
  • swap 出临时副本,再 merge 回原容器:std::vector tmp = std::move(vec1); std::merge(tmp.begin(), tmp.end(), vec2.begin(), vec2.end(), std::back_inserter(vec1));

最后提醒:STL 没有“in-place merge”标准接口c++23 仍未加入),std::inplace_merge 要求输入 range 是同一个容器中**连续的两段已排序子区间**,不适用于两个独立 vector 的合并场景。

text=ZqhQzanResources