erase(pos, len)用于删除string中指定位置和长度的子串,pos越界抛out_of_range异常,len超长则自动截断;常见错误是混淆len为结束下标或未检查pos合法性。

用 erase 删除 string 中指定位置和长度的子串
erase 是 std::string 的成员函数,最常用的方式是传入起始下标和要删除的字符数:str.erase(pos, len)。注意:如果 pos 超出当前字符串长度(str.length()),会抛出 std::out_of_range 异常;如果 len 超出剩余长度,它会自动截断到末尾,不会报错。
常见错误现象:写成 str.erase(i, j) 却误以为 j 是结束下标(实际是长度);或没检查 i 是否合法,导致崩溃。
- 要删从下标 2 开始、共 3 个字符:
str.erase(2, 3) - 要删从下标 5 到结尾的所有内容:
str.erase(5)(单参数版本,等价于erase(5, npos)) - 安全写法建议先判断:
if (pos
用迭代器范围删除子串(更通用但易错)
另一个重载是 erase(iterator first, iterator last),它删除 [first, last) 左闭右开区间内的字符。这个版本不依赖下标,适合配合 find、substr 等返回迭代器的操作使用,但要注意两个迭代器必须属于同一 string 对象,且 first 。
容易踩的坑:用 str.begin() + i 构造迭代器时,若 i > str.length(),行为未定义;删除后原迭代器失效,不能再用。
立即学习“C++免费学习笔记(深入)”;
- 删掉第一次出现的 "abc":
auto it = str.find("abc"); if (it != std::string::npos) str.erase(str.begin() + it, str.begin() + it + 3); - 删掉开头连续空格:
str.erase(str.begin(), std::find_if(str.begin(), str.end(), [](char c){ return !std::isspace(c); }));
删除所有匹配子串(不是 erase 一次能解决的)
erase 本身只删一次。想删掉所有出现的某个子串(比如全部 "xx"),得循环查找 + 删除,但要注意:正向遍历时,每次删除会导致后续内容前移,下标/迭代器需重新计算,否则跳过或重复处理。
推荐用反向查找(rfind)+ 反向删除,避免迭代器失效干扰;或者用 while 配合 find,每次更新查找起点为上一次删除位置之前。
- 安全删所有 "ab":
size_t pos = 0; while ((pos = str.find("ab", pos)) != std::string::npos) { str.erase(pos, 2); } - 更高效做法是构造新 string:
std::string res; res.reserve(str.length()); ...,尤其当删除频繁时,避免多次内存搬移
erase 后的 string 内存是否立即释放?
不会。erase 只修改逻辑长度(size()),底层缓冲区容量(capacity())通常不变。这意味着即使删光了整个 string,capacity() 还可能很大,后续追加仍无需分配。
如果你明确需要释放内存(例如处理完大字符串后长期驻留),得手动收缩:str.shrink_to_fit();(C++11 起),但它是非强制的,编译器可忽略;更可靠的是交换空 string:std::string().swap(str);。
这点在写长期运行的服务程序时容易被忽略——看似删了,内存却没还给系统。










