C++笔记九(STL与泛型编程)

本周内容
(1)迭代器的分类(category)
(2)迭代器分类对算法的影响
(3)STL算法
(4)仿函数/函数对象
(5)Adapter
(6)binder2nd and not1
(7)bind
(8)reverse_iterator、inserter、ostream_iterator、istream_iterator

  • 从语言层面讲:
    • 容器Container是个class template
    • 算法Algorithm是个function template
    • 迭代器Iterator是个class template
    • 仿函数Functor是个class template
    • 适配器Adapter是个class template
    • 分配器Allocator是个class template

Algorithm看不见Container,对其一无所知;所以,它所需要的一切信息都必须从Iterators取得,而Iterators(Containers供应)必须能够回答Algorithm的所有提问,才能搭配该Algorithm的所有操作。

一 迭代器的分类

//五种iterator category
struct input_iterator_tag{ };
struct output_iterator_tag{ };
struct forward_iterator_tag;public input_iterator_tag{ };
struct bidrectional_iterator_tag;public forward_iterator_tag{ };
struct random_access_iterator_tag;public bidrectional_iterator_tag{ };

五种iterator的关系如下:


iterator_category.png
  • 每个容器的iterator_category分别为:
    • array: random_access_iterator
    • vector: random_access_iterator
    • list: bidrectional_iterator
    • forward_list: forward_iterator
    • deque: random_access_iterator
    • set: bidrectional_iterator
    • map: bidrectional_iterator
    • multiset: bidrectional_iterator
    • multimap: bidrectional_iterator
    • unordered_set: forward_iterator
    • unordered_map: forward_iterator
    • unordered_multiset: forward_iterator
    • unordered_multimap: forward_iterator
    • istream: input_iterator
    • ostream: output_iterator

二 迭代器分类对算法的影响

  • 以copy为例,通过迭代器的分类和迭代器萃取机的分类,对于某些分类有特化版本,直接调用低阶动作函数速度极快,针对不同分类采取不同策略,这样对于代码效率提高是有好处的。


    iterator_category和type_traits对算法的影响.png
  • 算法源码中对iterator_category的暗示:通过写出一些明显的类型名称如下红色部分


    算法源码中对iterator_category的暗示.png

三 算法

  • qsort(快速排序),bsearch(二分法搜寻)这是C的函数;count_if(返回在[first, last)范围内满足特定条件的元素的数目),find(查找),sort(排序)是C++标准库提供的算法。
    因为他们接口满足如下:
template<typename Iterator>
std::Algorithm(Iterator itr1,Iterator itr2,...)
{
    ...
}
  • accumulate:用来计算特定范围内(包括连续的部分和初始值)所有元素的和,除此之外,还可以用指定的二进制操作来计算特定范围内的元素结果。
  • for_each:用于逐个遍历容器元素,它对迭代器区间[first,last)所指的每一个元素,执行由单参数函数对象f所定义的操作。
//range-based for statement since C++11
for( dec1:coll ) {
    statement
}
for( int i : {2,3,4,5,3,23,424,5} ){
    cout << i << endl;
}

  • replace:所有容器适用
    replace: 范围内所有等同于old_value者都以new_value取代
    replace_if:范围内所有满足pred()为true之元素都以new_value取代
    replace_copy:范围内所有等同于old_value者都以new_value放至新区间

  • count:用于统计某一值在一定范围内出现的次数
    count_if:返回在[first, last)范围内满足特定条件的元素的数目

    • 容器不带成员函数count():array,vector,list,forward_list,deque
    • 容器带有成员函数count():set/multiset,map/multimap,unordered_set/unordered_multiset,unordered_map/unordered_multimap
  • find:返回区间[first,end)中第一个值等于value的元素位置;若未找到,返回end。函数返回的是迭代器或指针,即位置信息。
    find_if:返回区间[first,end)中使一元判断式pred为true的第一个元素位置;若未找到,返回end。

    • 容器不带成员函数find():array,vector,list,forward_list,deque
    • 容器带有成员函数find():set/multiset,map/multimap,unordered_set/unordered_multiset,unordered_map/unordered_multimap
  • sort:对给定区间所有元素进行排序

    • 容器不带成员函数count():array,vector,deque, set/multiset,map/multimap,unordered_set/unordered_multiset,unordered_map/unordered_multimap
    • 容器带有成员函数count():list,forward_list
  • reverse iterator是一种反向遍历容器的迭代器


    reverse iterator.png
  • binary_search二分查找法,之前一定要先排序,调用的是lower_bound,与之相对应的有upper_bound,差别如下图:


    lower_bound and upper_bound.png

四 仿函数/函数对象

  • 仿函数或者说函数对象就是里面对()作重载的一种类。
  • 仿函数functors的可适配条件是,继承unary_function或者binary_function:


    functors可适配条件.png

五 Adapters

  • adapter需要回答算法提出的五个问题和仿函数提出的三个问题来确定参数类型,编译通过表示回答完毕。


    adapter.png

六 Binder2nd and not1

  • bind2nd辅助函数的价值是它是一个函数模板,会帮使用者作参数推导,比直接使用binder2nd需要知道operator的类型要方便。
  • 函数或者函数对象必须能够回答三个问题(上述已提),才能和adapter完美搭配。
  • typename的作用是当编译器运行到operation::second_argument_type时犹豫不决不知道什么类型时让它通过。


    binder2nd.png

    not1和bind2nd类似:


    not1.png

七 新型适配器bind(since C++11)

std::bind可以绑定:
(1)functions
(2)function objects
(3)member functions,_1(占位符号)必须是某个object地址。
(4)data members,_1必须是某个object地址。
返回一个function object ret。调用ret相当于调用上述1,2,3,或相当于取出4。

新版本代替了很多旧版本的东西,如下图:


bind.png

八 reverse_iterator、inserter、ostream_iterator、istream_iterator

下面介绍几种典型的适配器,这些适配器类似得对算法(例如copy)作了符号重载,使得同一个函数能够有不同的解读,这非常巧妙。

  • 下面两图是迭代器适配器:


    reverse_iterator.png
inserter.png
  • 下面两图是X(未知类型)适配器,:


    ostream_iterator.png
istream_iterator.png

下图是istream_iterator另一个例子,当使用者打出例2的代码,因为其实第一行已经有输入了,若是在下一行写入提示代码(“请输入”),则这一行提示代码不会出现,使得使用者会大吃一惊,需要注意!


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

推荐阅读更多精彩内容