【发布时间】:2023-03-17 21:35:02
【问题描述】:
当我使用时它是否正常工作(什么都不做)
vector<T> v;
v.erase(v.end());
我想使用类似的东西
v.erase(std::find(...));
我应该if还是v.end()?
C++.com 和 CPPreference 上没有关于它的信息
【问题讨论】:
标签: c++ vector iterator language-lawyer erase
当我使用时它是否正常工作(什么都不做)
vector<T> v;
v.erase(v.end());
我想使用类似的东西
v.erase(std::find(...));
我应该if还是v.end()?
C++.com 和 CPPreference 上没有关于它的信息
【问题讨论】:
标签: c++ vector iterator language-lawyer erase
标准并没有完全说明,但v.erase(q) 被定义为“擦除q 指向的元素”在[sequence.reqmts] 中。这意味着q 必须实际指向一个元素,而结束迭代器没有。传入 end 迭代器是未定义的行为。
不幸的是,你需要写:
auto it = std::find(...);
if (it != <the part of ... that specifies the end of the range searched>) {
v.erase(it);
}
当然,你可以定义:
template typename<Sequence, Iterator>
Iterator my_erase(Sequence &s, Iterator it) {
if (it == s.end()) return it;
return s.erase(it);
}
my_erase(v, std::find(v.begin(), v.end(), whatever));
关联容器上的c.erase() 返回void,因此要将这个模板推广到所有容器,您需要一些-> decltype 操作。
【讨论】:
position 定义为“指向单个元素的迭代器”。结束迭代器不指向单个元素。
end() - 1 而不是“向量结束”。
擦除end()(或者就此而言,甚至查看end() 的目标)是未定义的行为。未定义的行为允许有任何行为,包括在您的平台上“正常工作”。这并不意味着您应该这样做。它仍然是未定义的行为,当你以后最不期待的时候,我会以最糟糕的方式来咬你。
根据您的操作,您可能需要考虑使用set 或unordered_set,而不是此处的vector。
【讨论】:
{ auto it = v.find(x); if (it != x.end()) { v.erase(it); } }
end-1 当且仅当容器不为空时才会起作用。 (同pop_back)
你试过了吗?
v.erase(remove_if(v.begin(), v.end(), (<your criteria>)), v.end());
【讨论】:
remove_if