[C++再学习系列] STL容器删除操作总结

本文涉及的产品
容器镜像服务 ACR,镜像仓库100个 不限时长
简介:

由于容器所对应不同的迭代器、指针和引用的失效规则 ,使得容器的删除操作较为复杂。解决问题的最好方法取决于你是怎样鉴别出哪个对象是要被去掉的,储存它们的容器的类型,和当你删除它们的时候你还想要做什么(如果有的话)。为此分 3 种情况讨论:

去除一个容器中有特定值的所有对象:

 1) 如果容器是 vector 、 string 或 deque ,使用 erase-remove 惯用法。

2) 如果容器是 list ,使用 list::remove 。

3) 如果容器是标准关联容器,使用它的 erase 成员函数。 

示例:

Container c;  // 去除值为 1963 的元素

c.erase(remove(c.begin(), c.end(), 1963), c.end()); // 当 c 是 vector 、 string 或 deque 时, erase-remove 惯用法是去除特定值的元素的最佳方法

c.remove(1963);     // 当 c 是 list 时, remove 成员函数是去除特定值的元素的最佳方法

c.erase(1963);        // 当 c 为关联容器

去除一个容器中满足一个特定判定式的所有对象:

1) 如果容器是 vector 、 string 或 deque ,使用 erase-remove_if 惯用法。

2) 如果容器是 list ,使用 list::remove_if 。

3) 如果容器是标准关联容器,使用 remove_copy_if 和 swap( 使用临时容器 ) ,或写一个循环来遍历容器元素,当你把迭代器传给 erase 时记得后置递增它。 

示例:特定判断式 : bool badValue(int x);   // 返回 x 是否是“ bad ”

c.erase(remove_if(c.begin(), c.end(), badValue),  c.end()); // 当 c 是 vector 、 string 或 deque

c.remove_if(badValue);     // 当 c 是 list 时

// 关联容器的删除 使用  后置递增传给 erase 的迭代器  技术

AssocContainer c;

for (AssocContainer::iterator i = c.begin();  // for 循环的第三部分是空的

       i != c.end();  /*nothing*/ ){        // 循环体控制 i 递增

       if (badValue(*i)) {

c.erase(i++);   // 删除 i ,副作用 i 递增

    // logFile << "Erasing " << *i <<'\n';    // 写日志文件 ,or other operators

       }

       else ++i;         // 递增

}

说明:为什么要使用后置递增?后置递增会增加临时副本,效率低。但此处一定要使用后置递增。关联容器的 erase 并不返回删除后的下一个有效迭代器,同时删除后的迭代器是无效迭代器。 为达到删除单前迭代器同时指向后一个迭代器,我们必须在删除前递增迭代器,同时记录当前迭代器并删除。后置递增操作除增加迭代器外,同时返回递增前的对象拷贝,完全满足所需的删除条件。  

在循环内做某些事情并删除对象:

1) 如果容器是标准序列容器,写一个循环来遍历容器元素,每当调用erase时记得都用它的返回值更新你的迭代器。

2) 如果容器是标准关联容器,写一个循环来遍历容器元素,当把迭代器传给erase时记得后置递增它(例子同上) 。 

示例:标准序列容器 ( 标准关联容器,同上 )

for (SeqContainer::iterator i = c.begin();

       i != c.end();){

       if (badValue(*i)){

              logFile << "Erasing " << *i << '\n';

              i = c.erase(i);         // 通过把 erase 的返回值获取有效的迭代器

       }                                

       else

              ++i;

}

说明:对序列容器为什么不使用  后置递增你要传给 erase 的迭代器  技术?原因在于 vector 、 string 和 deque 容器调用erase 后,不仅使所有指向被删元素的迭代器失效,也使被删元素之后 的所有迭代器失效 ( 可能重新分配地址空间 ) 。为了使序列容器的 erase 后,能获得删除后的有效迭代器,其 erase 将迭代器作为返回值:一旦删除完成, erase 返回指向紧接在被删元素之后的元素的有效迭代器。

注意: list 虽是序列容器,但采用节点存储方式,故不存在地址重新分配。因此可以像 vector/string/deque 一样或像关联容器一样对待 list ;两种方法都可以为 list 工作。

---------------------------------------------------

兄弟的公司:立即购--手机购物,诚信网购

兄弟的公司:立即团

欢迎转载,请注明作者和出处

本文转自 zhenjing 博客园博客,原文链接:http://www.cnblogs.com/zhenjing/archive/2011/01/30/STL_remove.html    ,如需转载请自行联系原作者

相关文章
|
2月前
|
存储 算法 C++
C++ STL 初探:打开标准模板库的大门
C++ STL 初探:打开标准模板库的大门
112 10
|
5天前
|
算法 C语言 C++
【c++丨STL】list的使用
本文介绍了STL容器`list`的使用方法及其主要功能。`list`是一种双向链表结构,适用于频繁的插入和删除操作。文章详细讲解了`list`的构造函数、析构函数、赋值重载、迭代器、容量接口、元素访问接口、增删查改操作以及一些特有的操作接口如`splice`、`remove_if`、`unique`、`merge`、`sort`和`reverse`。通过示例代码,读者可以更好地理解如何使用这些接口。最后,作者总结了`list`的特点和适用场景,并预告了后续关于`list`模拟实现的文章。
22 7
|
23天前
|
存储 编译器 C语言
【c++丨STL】vector的使用
本文介绍了C++ STL中的`vector`容器,包括其基本概念、主要接口及其使用方法。`vector`是一种动态数组,能够根据需要自动调整大小,提供了丰富的操作接口,如增删查改等。文章详细解释了`vector`的构造函数、赋值运算符、容量接口、迭代器接口、元素访问接口以及一些常用的增删操作函数。最后,还展示了如何使用`vector`创建字符串数组,体现了`vector`在实际编程中的灵活性和实用性。
48 4
|
24天前
|
C语言 C++ 容器
【c++丨STL】string模拟实现(附源码)
本文详细介绍了如何模拟实现C++ STL中的`string`类,包括其构造函数、拷贝构造、赋值重载、析构函数等基本功能,以及字符串的插入、删除、查找、比较等操作。文章还展示了如何实现输入输出流操作符,使自定义的`string`类能够方便地与`cin`和`cout`配合使用。通过这些实现,读者不仅能加深对`string`类的理解,还能提升对C++编程技巧的掌握。
50 5
|
24天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
38 2
|
1月前
|
存储 算法 Linux
【c++】STL简介
本文介绍了C++标准模板库(STL)的基本概念、组成部分及学习方法,强调了STL在提高编程效率和代码复用性方面的重要性。文章详细解析了STL的六大组件:容器、算法、迭代器、仿函数、配接器和空间配置器,并提出了学习STL的三个层次,旨在帮助读者深入理解和掌握STL。
47 0
|
9天前
|
存储 编译器 C语言
【c++丨STL】vector模拟实现
本文深入探讨了 `vector` 的底层实现原理,并尝试模拟实现其结构及常用接口。首先介绍了 `vector` 的底层是动态顺序表,使用三个迭代器(指针)来维护数组,分别为 `start`、`finish` 和 `end_of_storage`。接着详细讲解了如何实现 `vector` 的各种构造函数、析构函数、容量接口、迭代器接口、插入和删除操作等。最后提供了完整的模拟实现代码,帮助读者更好地理解和掌握 `vector` 的实现细节。
18 0
|
26天前
|
存储 设计模式 C++
【C++】优先级队列(容器适配器)
本文介绍了C++ STL中的线性容器及其适配器,包括栈、队列和优先队列的设计与实现。详细解析了`deque`的特点和存储结构,以及如何利用`deque`实现栈、队列和优先队列。通过自定义命名空间和类模板,展示了如何模拟实现这些容器适配器,重点讲解了优先队列的内部机制,如堆的构建与维护方法。
32 0
|
2月前
|
存储 搜索推荐 C++
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器2
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器
59 2
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器2
|
2月前
|
存储 程序员 C++
C++常用基础知识—STL库(2)
C++常用基础知识—STL库(2)
79 5
下一篇
DataWorks