- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我需要从 std::vector 的中间移除元素。
所以我尝试了:
struct IsEven {
bool operator()(int ele)
{
return ele % 2 == 0;
}
};
int elements[] = {1, 2, 3, 4, 5, 6};
std::vector<int> ints(elements, elements+6);
std::vector<int>::iterator it = std::remove_if(ints.begin() + 2, ints.begin() + 4, IsEven());
ints.erase(it, ints.end());
在此之后,我希望 ints
vector 具有:[1, 2, 3, 5, 6]。
在 Visual Studio 2008 的调试器中,在 std::remove_if
行之后,修改了 ints
的元素,我猜我正在做某种事情这里有未定义的行为。
那么,如何从 vector 的范围中删除元素?
最佳答案
编辑:抱歉,这个的原始版本不正确。固定。
这是正在发生的事情。您对 remove_if
的输入是:
1 2 3 4 5 6
^ ^
begin end
并且 remove_if
算法查看 begin
和 end
之间的所有数字(包括 begin
,但不包括 end
), 并删除与您的谓词匹配的所有元素。所以在 remove_if
运行之后,你的 vector 看起来像这样
1 2 3 ? 5 6
^ ^
begin new_end
?
是一个我认为不是确定性的值,尽管如果保证它是任何值,它将是 4
。 new_end
是 std::remove_if 返回的内容,它指向您给它的输入序列的新结尾,现在删除了匹配元素
。请注意,std::remove_if
不会触及您提供的子序列之外的任何内容。对于更扩展的示例,这可能更有意义。
说这是你的输入:
1 2 3 4 5 6 7 8 9 10
^ ^
begin end
在 std::remove_if
之后,你得到:
1 2 3 5 7 ? ? 8 9 10
^ ^
begin new_end
想一想。它所做的是从子序列中删除 4 和 6,然后将子序列中的所有内容 向下移动以填充删除的元素,然后移动 end
迭代器到同一子序列的新末端。目标是满足其生成的 (begin
, new_end
] 序列与 (begin
, end
] 您传入的子序列,但删除了某些元素。您传入的 end
处或之后的任何内容都保持不变。
然后,您想要摆脱的是返回的结束迭代器和您给它的原始结束迭代器之间的所有内容。这些是 ?
“垃圾”值。所以你的删除调用实际上应该是:
ints.erase(it, ints.begin()+4);
您刚才调用的 erase
会删除执行删除操作的子序列末尾以外的所有内容,这不是您想要的。
让事情变得复杂的是 remove_if
算法实际上并没有在 vector 上调用 erase()
,也没有在任何时候改变 vector 的大小。它只是移动元素并在您要求它处理的子序列结束后留下一些“垃圾”元素。这看起来很愚蠢,但 STL 这样做的全部原因是为了避免 doublep 带来的无效迭代器的问题(并且能够在不是 STL 容器的东西上运行,比如原始数组)。
关于c++ - vector::erase 和 std::remove_if 的奇怪行为,其结束范围不同于 vector.end(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2717012/
假设您要按值从 vector 中删除单个元素。 remove 之间有什么区别? -删除: vector v; // add some values vector::iterator it = remo
class CSensor { public: CSensor(int nVal1,char* pVal2,unsigned int nVal3); CSensor(cons
在回答过程中another question我偶然发现 std::vector::erase() 和 std::deque::erase() 的措辞略有不同。 这就是 C++14 关于 std::de
我想处理 vector 中的元素一段时间。为了优化这一点,我不想在处理项目时删除它,而是在最后删除所有已处理的项目。 vector::iterator it; for(it = items.begin
是否 std::set erase ( x ) 在 STL 中以相同的方式实现 erase ( collection.find( x ) ); (其中 x 是 const key_type&) 吗?
This question already has answers here: How to catch .erase exception in vectors? (2个答案) 8天前关闭。 当我尝试
我有以下代码: #include using namespace std; int main() { set S; S.insert("item1"); S.insert("i
在'helper'函数中输出集合元素时,预计元素'2'已经被移除。但实际结果仍然是'1 2 3 4 5'。 但是,集合的大小是 4。 我想知道潜在的问题。 #include #include us
有点难以解释,但我想制作一个以图片为背景的网站,并在图片上覆盖一层灰色。然后我想在覆盖层上方添加一些文本,在文本所在的位置,灰色覆盖层被移除,因此文本是在没有灰色覆盖层的情况下写入的。 我在一年前用图
这是导致错误的代码: 工厂.h: #include #include namespace BaseSubsystems { template class CFactory
我正在研究结构 vector 。 当我试图用迭代器调用这个函数时,像这样: vec2.erase (vec2.begin()+iter2); 它向我发送了这个错误: "no match for 'op
进程文件: eraser or eraser.exe 进程名称: Eraser 进程类别:存在安全风险的进程 英文描述: eraser.exe is the main executable
我正在编写一个简单的程序,它使用了 std::map::erase . 该程序很好,但有一些我不明白的地方。 如果我传给 erase function 第一个迭代器超出第二个迭代器的间隔,该函数不会删
如果我的变量是 0 到 6294 之间的随机数,我如何让我的输出仅打印出第一个变量? 我的随机数代码是 int random = (int)(Math.random() * 6294); 例如,如果它
我正在尝试通过一个测试程序,并且我通过了所有测试,除了涉及到我的删除功能时,程序崩溃了。 我最好的猜测可能是前面或后面的哨兵节点正在被删除。或内存泄漏。 最佳答案 我怀疑有人愿意阅读所有这些代码并为您
背景:这是一个程序,用于将存储在 vector 中的数字逐位加 1。 前面有 0 的数字可以作为输入,但不能作为输出。例如:0123 和 123 都是有效输入 但 0124 是无效输出 而 124 是
我正在尝试创建一个垂直滚动的射击游戏,当您按下空格键时,会创建一颗子弹,然后当子弹离开屏幕时,子弹就会被销毁。我通过声明为 vector bullets; 的 vector 跟踪子弹当我试图销毁屏幕外
我在下面写这段代码,发现了这个奇怪的行为: #include #include #include using namespace std; int main() { map map1;
我为 SFML 编写了一个线程渲染器,它接受指向可绘制对象的指针并将它们存储在一个 vector 中以在每一帧中绘制。开始向 vector 添加对象和从 vector 中删除对象会经常导致段错误 (S
如果我想从 map 中删除单个元素(除了可能的错误检查之外我不关心返回值),有两种方法可以实现:erase按值键或删除: http://ideone.com/YWocN7 #include #inc
我是一名优秀的程序员,十分优秀!