c++中如何使用std::vector的assign方法_c++容器重新赋值技巧【汇总】

std::vector::assign 先清空原容器再用新值填充,size 由参数决定,capacity 可能收缩;常用重载为数量+值或迭代器范围,易因参数类型误判触发错误重载。

assign 会清空原 vector 并用新值填充

std::vector::assign 不是“追加”也不是“部分覆盖”,它先调用 clear(),再用指定内容重建整个容器。这意味着:无论原 vector 多大、有没有自定义分配器、是否预留了空间,调用后其 size() 完全由 assign 参数决定,capacity() 可能收缩(取决于实现,但通常不会保留多余容量)。

两种常用重载:数量+值 vs 迭代器范围

最常踩的坑是传错参数类型——比如把两个整数当成迭代器用,结果调用的是 assign(size_type n, const T&

t) 重载,而非你本意的区间赋值。

  • vec.assign(3, 42) → 赋值为 {42, 42, 42}(3 个 int 42)
  • vec.assign(it1, it2) → 用 [it1, it2) 区间内容替换整个 vec
  • 若误写 vec.assign(p, q)p, qint*,只要它们可转成整数(比如指针地址被隐式转成 size_t),就可能意外触发第一种重载,导致静默错误
std::vector v = {1, 2, 3};
v.assign(2, 99);           // v 变成 {99, 99}
auto it1 = v.begin();
auto it2 = v.begin() + 1;
v.assign(it1, it2);        // v 变成 {99} —— 注意:it1/it2 来自上一行的 v,不是原始 {1,2,3}

替代 assign 的更安全写法:swap + initializer_list 或 assign_range(C++23)

如果目标只是“用新数据完全替换”,直接构造新 vector 再 swap 更清晰、不易出错,且对移动语义友好的类型(如 std::string 元素)性能更好:

  • v = {10, 20, 30}; → 利用初始化列表赋值(C++11 起)
  • v.assign(std::begin(arr), std::end(arr)); → 数组转 vector,比 assign(n, val) 明确
  • C++23 起可用 v.assign_range(other_container);,语义更直白,且不依赖迭代器类型匹配

assign 后 capacity 可能变小,别依赖旧容量

很多代码假设 reserve() 后调用 assign() 仍保持原 capacity,但实际上标准不保证——assign 可能重新分配内存并释放多余空间。如果你后续要 push_back 大量元素,且已知最终大小,建议 assign 后显式 reserve()

std::vector v;
v.reserve(1000);
v.assign(500, "hello");  // v.capacity() 可能回落到 500,不一定是 1000
v.reserve(1000);         // 主动恢复,避免后续 push 导致多次 realloc

assign 的语义很干脆:扔掉旧的,换新的。但正因为它不保留历史容量、不区分“扩容”和“缩容”,在性能敏感路径中,是否调用它、何时调用它,得看你是真想“重置内容”,还是只是“更新一部分”。