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,q是int*,只要它们可转成整数(比如指针地址被隐式转成 size_t),就可能意外触发第一种重载,导致静默错误
std::vector<int> 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():
立即学习“C++免费学习笔记(深入)”;
std::vector<std::string> v; v.reserve(1000); v.assign(500, "hello"); // v.capacity() 可能回落到 500,不一定是 1000 v.reserve(1000); // 主动恢复,避免后续 push 导致多次 realloc
assign 的语义很干脆:扔掉旧的,换新的。但正因为它不保留历史容量、不区分“扩容”和“缩容”,在性能敏感路径中,是否调用它、何时调用它,得看你是真想“重置内容”,还是只是“更新一部分”。










