C++模板库笔记

C++标准模板库笔记(C++ Primer plus)

1.除序列外,vector还是可反转容器(reversible container)概念的模型。

它有两个类方法:rbegin()和rend(),前者返回一个指向反转序列的第

一个元素的迭代器,后者返回反转序列的超尾迭代器。这些迭代

器都是reverse_iterator,对这样的迭代器执行递增操作,将导致它

反向遍历可反转容器。

如果dice是一个vector<int>容器,而Show(int)是显示一个整数的函数,

则:

for_each(dice.begin(),dice.end(),Show); //正向显示dice的内容

for_each(dice.rbegin(),dice.rend(),Show);   //反向显示dice的内容

vector模板类是最简单的序列模型,除非其他类型的特殊有点能够更好地满足程序的要求,否则应该默认使用这种类型。

2.deque模板类(在deque头文件中声明)表示双端队列(double-ended queue),通常被简称为deque。

在STL中,其实现类似于vector容器,支持随机访问。其主要区别

在于,从deque对象的开始位置插入和删除元素的时间是固定的,

而不像vector中那样是线性时间的。

虽然vector和deque都提供对元素的随机访问和在序列中部执行线

性时间的插入和删除操作,但vector容器执行这些操作时速度要快些。

3.list模板类(在list头文件中声明)表示双向链表。

区别:list在链表中任一位置进行插入和删除的时间都是固定的,

因此,vector强调的是通过随机访问进行快速访问,而list强调的

是元素的快速插入和删除。

与vector相似,list也是可反转容器。不同的是,list不支持数组表

示法和随机访问,从容器中插入或删除元素之后,链表迭代器指

向元素将不变。(只是修改连接信息,指向某个元素的迭代器仍

然指向该元素。)

list 模板包含了链表专用的成员函数:
[图片上传失败...(image-ab7c24-1567059183323)]insert()和splice()之间的主要区别在于:insert()将原始区间的副本插入到目标地址,而splice()则将原始区间移到目标地址。移动后,被移动的容器将置空(splice()方法还有其他原型,用于移动单个元素和元素区间)。原始容器的迭代器将重新定位到新容器,迭代器仍然指向相同的元素。

1.void splice ( iterator position, list<T,Allocator>& x );

2.void splice ( iterator position, list<T,Allocator>& x, iterator it );

3.void splice ( iterator position, list<T,Allocator>& x, iterator first, iterator last );

position 是要操作的list对象的迭代器

1:会在position后把list&x所有的元素到剪接到要操作的list对象
2:只会把it的值剪接到要操作的list对象中
3:把first 到 last 剪接到要操作的list对象中

unique()只能将相邻的相同值压缩为单个值,但使用sort()后,再

使用unique()时,每个值将只占一个位置。

对于 非成员sort()函数,它需要随机访问迭代器,因为快速插入的

代价是放弃随机访问功能,所以不能将非成员函数sort()用于链表。

因此,list容器中包括了一个只能在类中使用的成员版本。

4.C++11新增了容器类forward_list,它实现了单链表。在这种链表

中,每个节点都只链接到下一个节点。因此,forward_list只需要

正向迭代器,是不可反转容器。相比于list,forward_list更简单,更

紧凑,但功能也更少。

5.queue模板类是一个适配器类

ostream_iterator模板也是一个适配器,让输出流能够使用迭代器

接口。queue模板让底层类(默认为deque)展示典型的队列接口。

queue模板的限制比deque更多,它不仅不允许随机访问队列元

素,甚至不允许遍历队列。

6.priority_queue模板类(在queue头文件中声明)是另一个适配

器类,它支持的操作与queue相同。两者的主要区别在于:在

priority_queue中,比重最大的元素被移到队首,内部区别在于,

默认的底层类是vector。
它可以修改用于确定哪个元素被放到队首的比较方式

7.stack 与queue相似,stack也是一个适配器类,它给底层类(默

认是vector)提供了典型的栈接口

stack模板的限制比vector更多,它不仅不允许随机访问栈元素,

甚至不允许遍历栈。

8.array 模板类array在头文件array中定义,它不是STL容器,因为

其长度是固定的。像push_back(),insert()这种调整容器大小的操

作,array都没有,但是它有operator和at()。可将很多标准STL算

法用于array对象,如copy()和for_each()。

9.关联容器(associative container) 将值与键关联在一起,并用

键来查找值。

关联容器的优点在于,它提供了对元素的快速访问。关联容器允

许插入新元素,但不能指定元素的插入位置。原因是关联容器通

常有用于确定数据放置位置的算法,以便能快速检索信息。

关联容器通常是使用某种树实现的。像链表一样,树的节点使得

添加或删除数据项比较简单。但相对于链表,树的查找速度更

快。

STL提供了四种关联容器:set, multiset, map, 和multimap
(set和map头文件)

最简单的关联容器是set,其值和键的类型相同,键是唯一的。

multiset类似于set,只是可能有多个值的键相同。

map中 值与键的类型不同,键是唯一的。但是multimap中一个键

可以对应多个值。

10.无序关联容器 与关联容器一样,也将值与键关联起来,并使用

键来查找值。但底层的差别在于,关联容器是基于树结构的,而

无序关联容器是基于哈希表的,目的是提高元素的添加和删除速

度以及提高查找算法的效率。

11.函数对象: 很多STL算法都使用函数对象——也叫函数符

(functor)。函数符是可以以函数方式与()结合使用的任意对象。这

包括函数名,指向函数的指针和重载了()运算符的类对象。

12.对于所有内置的算术运算符,关系运算符和逻辑运算符,STL

都提供了等价的函数符

2
13. STL算法有两个主要的通用部分。首先,它们都使用模板来提

供泛型;其次它们都使用迭代器来提供访问容器中数据的通用表

示。

STL将算法库分成4种:

  • 非修改式序列操作(find(),for_each()...)
  • 修改式序列操作(transform(),random_shuffle(),copy()...)
  • 排序和相关操作(sort(),其它集合操作等等)
  • 通用数字运算(比如将区间的内容累积,计算两个容器的内部乘积)
    前3组在头文件algorithm中,第4组在numeric中。

有些算法有两个版本:就地版本和复制版本。STL的约定是,复

制版本的名称将以_copy结尾。复制版本将接受一个额外的输出迭

代器参数,该参数指定结果的放置位置。如:void replace_copy(...)

有些函数有这样的版本,即根据将函数应用于容器元素得到的结果

来执行操作。这些版本的名称通常以_if结尾。如:void replace_if(...)

14.string类虽然不是STL的组成部分,但设计它时考虑到了STL,

因此可以使用STL接口。

next_permutation()算法将区间内容转换为下一种排列方式。对于

字符串,排列按照字母递增的顺序进行。如果成功则返回true,

如果区间已经处于最后的序列中,则返回false。比如:

string letters;
...
sort(letters.begin(),letters.end());
while(next_permutation(letters.begin(),letters.end())
cout<<letters<<endl;

这个将输出字符串的所有排列。

15.有时可以选择使用STL方法STL函数。通常 方法 是更好的选

择。首先,它更适合于特定的容器;其次,作为成员函数,它可

以使用模板类的内存管理工具,从而在需要时调整容器的长度。

而函数,它不是成员,并不能调整容器的长度。(不过可以让函

数的返回值是指向新的超尾值的迭代器,来提醒容器长度改变了)

16.编写一个程序,让用户输入单词。希望最后得到一个按输入顺

序排列的单词列表、一个按字母顺序排列的单词列表(忽略大小写)

,并记录每个单词被输入的次数。

//set会自动对元素进行排序,并且不会有相同元素出现。
#include<iostream>
#include<string>
#include<vector>
#include<set>
#include<map>
#include<iterator>
#include<algorithm>
using namespace std;

char toLower(char ch) { return tolower(ch); }
string& ToLower(string& st);
void display(const string& s);

int main()
{
    vector<string>words;
    cout << "Enter words (enter quit to quit):" << endl;
    string input;
    while (cin >> input && input != "quit") {
        words.push_back(input);
    }
    cout << "You entered the following words:" << endl;
    for_each(words.begin(), words.end(), display);
    cout << endl;
    
    set<string>wordset;
    transform(words.begin(), words.end(), 
        insert_iterator<set<string>>(wordset, wordset.begin())
        , ToLower);
    cout << endl << "Alphabetic list of words:" << endl;
    for_each(wordset.begin(), wordset.end(), display);
    cout << endl;

    map<string, int>wordmap;
    set<string>::iterator si;
    for (si = wordset.begin(); si != wordset.end(); ++si)
        wordmap[*si] = count(words.begin(), words.end(), *si);

    cout << endl << "Word frequency:" << endl;
    for (si = wordset.begin(); si != wordset.end(); ++si)
        cout << *si << ": " << wordmap[*si] << endl;
    return 0;
}

string& ToLower(string& st)
{
    transform(st.begin(), st.end(), st.begin(), toLower);
    return st;
}

void display(const string& s)
{
    cout << s << " ";
}

C++提供三个数组模板:vector、valarray和array。
vector模板类支持面向容器的操作,
valarray是面向数值计算的,不是STL的一部分,
array是为替代内置数组而设计的。
若将两个数组各个元素相加,并将和存储到新的的对象中:
vector< double > ved1(10),ved2(10),ved3(10);
array <double,10>vod1,vod2,vod3;
valarray< double >vad1(10),vad2(10),vad3(10);

对于vector类:
transform(ved1.begin(),ved1.end(),ved2.begin(),ved3.begin(),plus< double >());
对于array类:
transform(vod1.begin(),vod1.end(),vod2.begin(),vod3.begin(),plus< double >());
对于valarray类:
vad3=vad1+vad2; //因为valarray类重载了所有算术运算符。

显然,执行多步运算时,valarray接口的简单性更加明显。

17.initializer_list
使用initializer_list对象,必须包含头文件initializer_list。这个模板

类包含成员函数begin()和end(),可以使用这两个函数来访问列表

元素。它还包含成员函数size(),返回元素数。

可按值或者按引用传递initializer_list对象。这种对象本身很小,通

常是两个指针(一个指向开头,一个指向末尾的下一个元素),

也可能是一个指针和一个表示元素数的整数,所以传递方式对性

能的影响不大。

initializer_list的迭代器类型为const。

initializer_list类的初衷是将一系列值传递给构造函数或其他函数。

18.STL是一个容器类模板,迭代器模板,函数对象模板和算法模

板的集合,它们的设计都是基于泛型编程原则。算法通过使用模

板,从而独立于所存储的对象的类型;通过使用迭代器接口,从

而独立于容器的类型。

19.智能指针
智能指针模板(auto_ptr,unique_ptr和shared_ptr)(头文件

memory)都定义了类似指针的对象,可以将new获得(直接或间

接)的地址赋给这种对象。当智能指针过期时,其析构函数将使

用delete来释放内存。

所有智能指针类都有一个explicit构造函数,该构造函数将指针作

为参数。

三种智能指针都应该避免的一点:

double a=1.0;
shared_ptr<double> pa(&a);  //NO!
//因为这样会导致delete运算符用于非堆内存。

为了避免删除同一个对象2次,可以用下面这些方法:

  • 重载赋值运算符,使之执行深复制
  • 建立所有权(ownership)概念,对于特定的对象,只能有一个智能指针拥有它,然后让赋值操作转让所有权。(这种方法的缺点是当指针丧失所有权后,若我们继续用这个指针去访问对象,会引发异常,因为丧失所有权后的智能指针是空指针。)
  • 引用计数(reference counting),跟踪引用特定对象的智能指针数。 比如赋值时,计数加1,指针过期时,计数减1,当最后一个指针过期时,调用delete。

unique_ptr和auto_ptr都是用建立所有权的办法,但是unique_ptr比auto_ptr更严谨,不容易引发异常。假如程序试图将一个unique_ptr赋给另一个时,如果源unique_ptr是个临时右值(比如返回值是非引用变量),编译器不会报错,但是如果源unique_ptr将存在一段时间,编译器会报错。

20.相比于auto_ptr,unique_ptr还有另一个优点:
因为delete和new配对,delete[]和new[]配对。而模板auto_ptr使用delete而不是delete[],因此只能与new一起使用,而不能与new[]一起使用。但unique_ptr可以(shared_ptr也不可以):
std::unique_ptr<double[]>pda(new double(5))

21.注意下面show函数里面的参数


3
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,776评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,527评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,361评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,430评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,511评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,544评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,561评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,315评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,763评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,070评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,235评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,911评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,554评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,173评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,424评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,106评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,103评论 2 352

推荐阅读更多精彩内容