![](/img/trans.png)
[英]string::replace throws std::out_of_range on valid iterators
[英]Replace in string iteration (out_of_range)
我编写了一个对字符串进行百分比编码的函数,如下所示:
string percent_encode(string str)
{
string reserved =
// gen-delims
":/?#[]@"
// sub-delims
"!$&'()*+,;="
;
for(string::iterator i = str.begin(); i < str.end(); i++) {
int c = *i;
// replaces reserved, unreserved non-ascii and space characters.
if(c > 127 || c == 32 || reserved.find(*i) != string::npos) {
std::stringstream ss;
ss << std::hex << c;
str.replace(i, i + 1, "%" + ss.str());
}
}
return str;
}
当我为“ a&b”之类的字符串调用此函数时,将抛出out_of_range异常:
terminate called after throwing an instance of 'std::out_of_range'
what(): basic_string::replace
我使用调试器跟踪了此异常,发现替换效果很好,但是它以某种方式迭代了end();
这是我看到迭代器“ i”时得到的:
{_M_current = 0x7fc43d61bd78 "a&b"}
{_M_current = 0x7fc43d61bd79 "&b"}
{_M_current = 0x7fc43d61bd7a "b"}
{_M_current = 0x7fc43d61bd7b ""}
{_M_current = 0x7fc43d61bd7c "o = a&b\n"}
{_M_current = 0x7fc43d61bd7d " = a&b\n"}
然后,它尝试替换“ =”,并以out_of_range异常失败。 我不明白,迭代器如何有可能明显超出end()。
如果有人可以向我解释,我将不胜感激,因为我无法在网络上找到遇到相同问题的人。
谢谢并恭祝安康,
复活
编辑:
啊,我真的觉得很复杂。 x)这就是我现在解决的方式。
string percent_encode(string str)
{
string reserved =
// gen-delims
":/?#[]@"
// sub-delims
"!$&'()*+,;="
;
std::stringstream ss;
for(string::iterator i = str.begin(); i < str.end(); i++) {
// encodes reserved, unreserved non-ascii and space characters.
int c = *i;
if(c > 126 || c == 32 || reserved.find(*i) != string::npos) {
ss << '%' << std::hex << c;
} else {
ss << *i;
}
}
return ss.str();
}
谢谢迭戈:)
replace
使当前的迭代器无效,因此它可能会超出结尾。
有几种正确编写此代码的方法。 例如,生成(并返回)新字符串会更容易,甚至可能更有效(请注意,replace也必须将字符串的其余部分移动一个位置)。 同样,使用更新的字符串长度和带有索引的位置进行播放。
但是 ,我想到的最好是返回一个全新的字符串。 功能更多:)
问题是您在迭代str时更改了str。 更改字符串的内容时,迭代器无效。 解决方案是使用包含转换结果的字符串的另一个副本。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.