0.前言
随机访问迭代器: vector、string、deque
STL的一个革命性的方面就是它的计算复杂性保证

条款01:慎重选择容器类型

c++提供的容器:
标准STL序列容器:vector、string、deque、list
标准STL关联容器:set、multiset、map、multimap
非标准STL容器:stack、queue、hash_set、hash_map

STL容器的一个分类方法:
连续内存容器:逻辑相邻的元素物理地址也相邻:vector、string、deque
基于节点的容器:逻辑相邻的元素的物理地址一般不相邻:list、set、map

条款02:不要试图编写独立于容器类型的代码

STL以泛化原则为基础:
数组被泛化为"以其包含的对象的类型为参数"的容器
指针被泛化为"以其指向的对象的类型为参数"的迭代器
函数被泛化为"以其使用的迭代器的类型为参数"的算法

条款03:确保容器中的对象拷贝正确而高效

当通过 insert、push_back 向容器中加入对象时,存入容器的是你所指定对象的拷贝
当通过 front、back 从容器中取出一个对象时,你所得到的是容器中所保存对象的拷贝

使拷贝动作高效、正确,并防止派生类对象剥离问题的一个简单办法就是:使容器包含指针而不是对象(智能指针)

void func() {
typedef std::tr1::shared_ptr<Widget> SPW;
std::vector<SPW> vpw;
for (int i = ; i < ; ++i)
vpw.push_back(SPW(new Widget)); //无内存泄露且异常安全
}
条款04:调用 empty 而不是 检查 size() 是否为0

if (c.size() == 0) 本质上与 if (c.empty()) 等价
我们应该总是调用 empty(),因为 empty() 操作对所有的标准STL容器都是O(1),size() 对某一些list容器是O(n) (不同平台list实现略有不同)

条款05:区间成员函数优先于与之对应的单元素成员函数

区间成员函数:使用两个迭代器参数来确定该成员操作所执行的区间

std::vector<int> vec1;
std::vector<int> vec2;

现在让vec1的整体与vec2 的后半部分相同:

(1)区间成员assign版本:

vec1.assign(vec2.begin() + vec2.size() /  , v2.end());

//注:assign() 对所有的标准序列容器 vector、string、deque、list都适用,当需要完全替换一个容器的内容时,应该优先使用assign()

(2)单元素循环版本:

vec1.clear();
for (auto it = vec2.begin() + vec2.size() / ; it != vec2.end(); ++it)
vec1.push_back(*it);

(3)区间成员insert版本:

vec1.clear();
vec1.insert(vec1.end(), vec2.begin() + vec2.size() / , vec2.end());

假定把一个int数组拷贝到一个vector里面:

方法一:区间成员函数

int data[numValues];
vector<int> vec;
vec.insert(vec.begin(), data, data + numValues);

方法二:单元素循环

vector<int>::iterator it = vec.begin();
for (int i = ; i < numValues; ++i) {
it = vec.insert(it, data[i]);
++it;
}

注:任何push和insert操作都可能导致迭代器失效,当元素插入到vector、deque、string时,必须确保迭代器在每次循环后都得到更新

a.方法一之调用一次insert函数,方法二调用了numValues次insert函数
b.假设vec插入前已有n个元素,则方法二总共会有 n*numValues 次元素移动,如果元素为自定义类型,则会调用 n*numValues次赋值操作
方法一中,c++标准要求区间insert函数把现有容器中的元素直接移动到他们的最终位置,总共会有 n 次元素移动
c.方法二使用区间insert可以在元素插入之前计算需要多少新内存,从而只进行一次 vector的内存扩充操作
方法一单元素循环插入numValues次最多可导致log(numValues)次新内存分配(涉及到新内存分配、元素拷贝到新内存、旧内存回收)

区间函数:
(1)区间创建:所有标准容器提供如下形式的构造函数

container::container(InputIterator begin, InputIterator end);

(2)区间插入:所有标准序列容器提供如下形式的 insert

container::insert(iterator pos, InputIterator begin, InputIterator end);//元素在pos之前插入

所有标准关联容器提供如下形式的insert:

container::insert(InputIterator begin, InputIterator end);//关联容器使用比较函数来决定元素该插入何处

(3)区间删除:所有标准容器提供如下形式的 erase

container::erase(iterator begin, iterator end);

(4)区间赋值:所有标准序列容器都提供如下形式的 assign

container::assign(InputIterator begin, InputIterator end);

总结:应该优先使用区间成员函数而不是单元素循环版本,因为使用区间可以预先知道操作结果,从而一次性移动到最终位置

条款06:当心c++编译器最烦人的分析机制

当区间创建函数 container::container(InputIterator begin, InputIterator end) 使用的迭代器是 istream_iterator时可能遇到c++最烦人的分析机制

条款07:容器中包含 new 创建的指针,必须在对象析构之前 delete 掉(注:采用 shared_ptr,避免人肉 delete)
条款08:采用shared_ptr智能指针,禁止使用auto_ptr
条款09:慎重选择删除元素的方法

假如有以下标准STL容器: container<int> c; 现在要删除其中所有值为2012的元素
如果是连续内存容器(vector、string、deque):

c.erase(remove(c.begin(), c.end(), ), c.end()); //erase-remove是删除特定值的最好办法

如果是list容器:

c.remove(c.begin(), c.end(), );//remove是list容器删除特定值的最好办法

如果是标准关联容器(set、map):

c.erase();//erase是set、map容器删除特定值的最好办法(无remove成员函数)

对于 vector、string、deque、list 删除满足条件的特定值时,采用 remove_if

而对于关联容器,比如set容器:写一个循环遍历容器中的元素

std::set<int> c;
for (auto it = c.begin(); it != c.end(); ++it) {
if (func(*it)) c.erase(it);//错误!! 如果erase(*it)之后,位于it之后的所有迭代器全部失效,而循环变量it不能被递增
}

正确写法:

for (auto it = c.begin(); it != c.end(); ) {
if (func(*it)) {
c.erase(it++);
} else {
++it;
}
}
条款10:了解分配子(allocator)的约定和限制
条款11:理解自定义分配子的合理用法
条款12:切勿对STL容器的线程安全性有不切实际的依赖
vector<int> vec;                                            //第1行
vector<int>::iterator it(find(vec.begin(), vec.end(), )); //第2行
if (it != vec.end()) *it = ; //第3行

在多线程环境中,可能在第1行刚刚完成之后,另一个不同的线程会更改vec(扩充重分配内存),这样导致 it 迭代器失效,对it解引用错误
此时应该对这3行代码块进行手工同步控制,引入互斥锁(以对象管理锁资源)

STL学习笔记(一) 容器的更多相关文章

  1. STL学习笔记— —无序容器(Unordered Container)

    简单介绍 在头文件<unordered_set>和<unordered_map> 中定义 namespace std { template <typename T, ty ...

  2. STL学习笔记--特殊容器

    容器配接器 (1) stack 栈 后进先出(LIFO), 头文件#include<stack> template<class _Ty, class _Container = deq ...

  3. STL学习笔记--各种容器的运用时机

    如何选择最佳的容器类别? 缺省情况下应该使用vector.vector的内部结构简单,并允许随机存取,所以数据的存取十分方便灵活,数据的处理也够快. 如果经常要在序列的头部和尾部安插和移除元素,应采用 ...

  4. Effective STL 学习笔记 32 ~ 33

    Effective STL 学习笔记 32 ~ 33 */--> div.org-src-container { font-size: 85%; font-family: monospace; ...

  5. Effective STL 学习笔记 31:排序算法

    Effective STL 学习笔记 31:排序算法 */--> div.org-src-container { font-size: 85%; font-family: monospace; ...

  6. Effective STL 学习笔记 Item 30: 保证目标区间足够大

    Effective STL 学习笔记 Item 30: 保证目标区间足够大 */--> div.org-src-container { font-size: 85%; font-family: ...

  7. Effective STL 学习笔记: Item 22 ~ 24

    Effective STL 学习笔记: Item 22 ~ 24 */--> div.org-src-container { font-size: 85%; font-family: monos ...

  8. Effective STL 学习笔记 Item 21:Comparison Function 相关

    Effective STL 学习笔记 Item 21:Comparison Function 相关 */--> div.org-src-container { font-size: 85%; f ...

  9. Effective STL 学习笔记:19 ~ 20

    Effective STL 学习笔记:19 ~ 20 */--> div.org-src-container { font-size: 85%; font-family: monospace; ...

随机推荐

  1. Android(java)学习笔记148:网易新闻RSS客户端应用编写逻辑过程

    1.我们的项目需求是编写一个新闻RSS浏览器,RSS(Really Simple Syndication)是一种描述和同步网站内容的格式,是使用最广泛的XML应用.RSS目前广泛用于网上新闻频道,bl ...

  2. Solr笔记(2)_Schema.xml和solrconfig.xml分析

    现在我们开始研究载入的数据部分(importing data) 在正式开始前,我们先介绍一个存储了大量音乐媒体的网站http://musicbrainz.org , 这里的数据都是免费的,一个大型开放 ...

  3. 动态规划初步--最长上升子序列(LIS)

    一.问题 有一个长为n的数列 a0,a1,a2...,an-1a.请求出这个序列中最长的上升子序列的长度和对应的子序列.上升子序列指的是对任意的i < j都满足ai < aj的子序列. 二 ...

  4. NLP.TM | GloVe模型及其Python实现

    在进行自然语言处理中,需要对文章的中的语义进行分析,于是迫切需要一些模型去描述词汇的含义,很多人可能都知道word2vector算法,诚然,word2vector是一个非常优秀的算法,并且被广泛运用, ...

  5. shell脚本,awk合并一列的问题。

    文件 file2内容如下:0 qwert1 asdfghjk2 asdjkl2 zxcvbn3 dfghjkll4 222224 tyuiop4 bnm 让第一列相等的合并成一行,不要第一列,也就是变 ...

  6. ZOJ Monthly, January 2019-Little Sub and Pascal's Triangle

    这个题的话,它每行奇数的个数等于该行行号,如果是0开始的,就该数的二进制中的1的个数,设为k,以它作为次数,2k就是了. #include <stdio.h> int main() { i ...

  7. Django REST framework 的功能

    1. 认证Authentication 方法一:在配置文件中配置全局默认的认证方案 REST_FRAMEWORK = { 'DEFAULT_AUTHENTICATION_CLASSES': ( 're ...

  8. 玩转ApplicationContextAware

    当一个类实现了这个接口之后,这个类就可以方便地获得 ApplicationContext 中的所有bean.换句话说,就是这个类可以直接获取Spring配置文件中,所有有引用到的bean对象.结合工厂 ...

  9. [转]Makefile中的wildcard/notdir/patsubst

    1.wildcard : 扩展通配符 2.notdir : 去除路径 3.patsubst :替换通配符 例子:建立一个测试目录,在测试目录下建立一个名为sub的子目录$ mkdir test$ cd ...

  10. shell相关指令介绍$*和$#以及$?和if [[ ! -z $1 ]]

    $#,脚本运行时后跟的参数个数 #! /bin/bash case "$#" in 0) printf "Enter a number: " read n=$R ...