注意事项 :

  • 实际使用的时候,使用的是<algorithm>这个头文件,不是题目中的< stl_algobase.h >

equal函数

  • 如果两个序列在[firsLlast) 区间内相等,equal() 返 回 true.如果第二序列的元素比较多,多出来的元素不予考虑。因此,如果我们希望保证两个序列完全相等,必须先判断其元素个数是否相同
    int ia[9] = {0,1,2,3,4,5,6,7,8};std::vector<int> iv1(ia,ia+5);std::vector<int> iv2(ia,ia+9);if(iv1.size() == iv2.size() &&std::equal(iv1.begin(),iv1.end(),iv2.begin())){}
  • 抑或使用容器所提供的equality操作符,例如 vecl = =vec2.如果第二序列的 元素比第一序列少,这个算法内部进行迭代行为时,会超越序列的尾端,造成不可预测的结果
  • 第一版本缺省采用元素型别所提供的equality操作符来进行大小比 较,第二版本允许我们指定仿函数pred作为比较依据

fill

  • 将 [first/ last) 内的所有元素改填新值.

    int ia[9] = {0,1,2,3,4,5,6,7,8};std::vector<int> iv1(ia,ia+5);std::vector<int> iv2(ia,ia+9);std::fill(iv2.begin(),iv2.end(),-1);for (auto elem:iv2) {std::cout<< elem << ' ';}std::cout<< "\n";

fill_n

  • 将[firsllast)内的前n 个元素改填新值,返回的迭代器指向被填入的最后一个元素的下一位置。

  • 如果n 超越了容器的现有大小,会造成什么结果?例如

  • 我们很容易就可以从源代码知道,由于每次迭代进行的是assignment操作,是一种覆写(overwrite)操作,所以一旦操作区间超越了容器 大小,就会造成不可预期的结果。解决办法之一是,利用inserter ()产生一个具 有插入(insert)而非覆写(overwrite)能力的迭代器。
  • inserter( ) 可产生一个用来修饰迭代器的配接器(iterator adapter)。用法如下
int main(int argc,char* argv[]) {int ia[9] = {0,1,2};std::vector<int> iv1(ia,ia+3);std::fill(iv1.begin(),iv1.end(),-1);std::fill_n(iv1.begin(),5,7);for (auto elem:iv1) {std::cout<< elem << ' ';}std::cout<< "\n";
}
  • 但是实际使用的时候 并没有出现问题

iter_swap

  • iter_swap( ) 是 “迭代器之valuetype派上用场的一个好例子。是的,该函数 必须知道迭代器的veluetype,才能够据此声明一个对象,用来暂时存放迭代器所 指对象。为此,上述源代码特别设计了一个双层构造,第一层调用第二层,并多出一个额外的参数value_type(a) 这么一来,第二层就有VClIue type可以用了。
  • 乍见之下你可能会对这个额外参数在调用端和接受端的型别感到讶异,调用端是value_type (a), 接受端却是T*。只要找出value_type ()的定义瞧瞧,就一点也不奇怪了:
#include <iostream>
#include <algorithm>
#include <vector>
#include <random>
#include <functional>template<class ForwardIt>
void selection_sort(ForwardIt begin,ForwardIt end){for (ForwardIt i = begin;i!=end;++i) {std::iter_swap(i,std::min_element(i,end));}
}
int main(int argc,char* argv[]) {std::random_device rd{};std::mt19937 gen(rd());std::uniform_int_distribution<>dist(-9,9);std::vector<int>v{};std::generate_n(std::back_inserter(v),20,std::bind(dist,gen));std::cout << "Before sort:" << std::showbase;for (auto e:v) {std::cout<<e<<' ';}selection_sort(v.begin(),v.end());std::cout<<"\nAfter sort:";for (auto e:v) {std::cout << e << ' ';}}

lexicographic al_.com pare

  • 以 “字典排列方式”对两个序列[firstlAastl)和 [first2,last2)itt行比较。比较操作针对两序列中的对应位置上的元素进行,并持续直到(1 )某一组对应元素彼此不相等;(2 )同时到达last1 和 last2 (当两序列的大小相同);⑶ 到 达 lastl或last2 (当 两 序 列 的 大 小 不 同 )。
  • 第一序列以字典排列方式(lexicographically)而言不小于第二序列

  • 第二版本允许你指定一个仿函数comp作为比较操作之用,取代元素型别所提 供 的 less-than (小 于 )操 作 符
#include <iostream>
#include <algorithm>
#include <vector>
#include <random>int main(int argc,char* argv[]) {std::vector<char>v1{'a','b','c','d'};std::vector<char>v2{'a','b','c','d'};std::mt19937 g{std::random_device{}()};while (!std::lexicographical_compare(v1.begin(),v1.end(),v2.begin(),v2.end())){for (auto c:v1) std::cout << c << ' ';std::cout << ">= ";for (auto c:v2) std::cout << c << ' ';std::cout << "\n";std::shuffle(v1.begin(),v1.end(),g);std::shuffle(v2.begin(),v2.end(),g);}for (auto c:v1) std::cout << c << ' ';std::cout << "< ";for (auto c:v2) std::cout << c << ' ';std::cout << "\n";
}

mism atch

  • 用来平行比较两个序列,指出两者之间的第一个不匹配点。返回一对迭代器;分别指向两序列中的不匹配点,如下图。
  • 如果两序列的所有对应元素都匹配,返回的便是两序列各自的last迭代器。
  • 缺省情况下是以equality操作符来比较元素; 但第二版本允许用户指定比较操作。如果第二序列的元素个数比第一序列多,多出来的元素忽略不计。如果第二序列的元素个数比第一序列少,会发生未可预期的行为。

#include <iostream>
#include <algorithm>
#include <vector>
#include <random>bool mypredicate(int i,int j){return (i==j);
}int main(int argc,char* argv[]) {std::vector<int>my_vector{};for (int i = 1; i < 6; ++i) {my_vector.emplace_back(i*10);//10 20 30 40 50 }int myints[] = {10,20,80,320,1024};std::pair<std::vector<int>::iterator,int*>mypair;mypair = std::mismatch(my_vector.begin(),my_vector.end(),myints);std::cout << "First mismatching elements: " << *mypair.first;std::cout << " and " << *mypair.second << std::endl;++mypair.first;++mypair.second;mypair = std::mismatch(mypair.first,my_vector.end(),mypair.second, mypredicate);std::cout << "Second mismatching elements: " << *mypair.first;std::cout << " and " << *mypair.second << std::endl;return 0;
}

copy----- 强化效率无所不用其极

  • 不论是对客端程序或对STL内部而言, copy() 都是一个常常被调用的函数。由 于 copy进行的是复制操作,而复制操作不外乎运用assignment operator或 copy constructor (copy算法用的是前者),但是某些元素型别拥有的是trivial assignment operator, 因此,如果能够使用内存直接复制行为(例如C 标准函数memmove或 memcpy) , 便能够节省大量时间。为此,SGI STL的 copy 算法用尽各种办法,包括函数重载(function overloading) 、型别特性(type traits)、偏特 化 (partial specialization) 等编程技巧,无所不用其极地加强效率。图 6-2表ZK整个copy() 操作的脉络。配合稍后出现的源代码,可收一目了然之效。

  • copy算法可将输入区间[first, last)内的兀素复制到输出区间[result, result+(last-first)) 内• 也就是说,它会执行赋值操作 *result = *first, * (result + 1) = * ( f irst + 1) , … 依 此 类 推 。 返 回 一 个 迭 代 器 : result+(last-first)= copy 对其template参数所要求的条件非常宽松。其输入区间只需由Inputiterators构成即可,输出区间只需由Outputiterator构成即可。
  • 这意味着你可以使用copy算法,将任何容器的任何一段区间的内容,复制到任何 容器的任何一段区间上

  • 如果输入区间和输出区间完全没有重叠,当然毫无问题,否则便需特别注意•为什么图6-3第二种情况(可能)会产生错误?从稍后即将显示的源代码可知,copy 算法是一一进行元素的赋值操作,如果输出区间的起点位于输入区间内,copy算法便(可能)会在输入区间的(某些)元素尚未被复制之前,就覆盖其值,导致错误结果.在这里我一再使用“可能”这个字眼,是因为,如果copy算法根据其所 接收的迭代器的特性决定调用 memmove() 来执行任务,就不会造成上述错误,因 为 memmove() 会先将整个输人区间的内容复制下来,没有被覆盖的危险。
    int ia[] = {0,1,2,3,4,5,6,7,8};//输出区间的终点和输入的区间重叠  没有问题std::copy(ia+2,ia+7,ia);std::for_each(ia,ia+9,display<int>());std::cout << "\n";
    // 以下,输出区间的起点与输入区间重叠,可能会有问题int ia[] = {0,1,2,3,4,5,6,7,8};std::copy(ia+2,ia+7,ia+4);std::for_each(ia,ia+9,display<int>());std::cout << "\n"; //0 1 2 3 2 3 4 5 6
    int ia[] = {0,1,2,3,4,5,6,7,8};//deque 拥有 RandomAcCGSSltGrCltorstd::deque<int>id(ia,ia+9);std::deque<int>::iterator first = id.begin();std::deque<int>::iterator last = id.end();++++first;  //advance(first,2);std::cout << *first << std::endl; //2----last;   //advance(last,-2);std::cout << *last << std::endl;  //7std::deque<int>::iterator result = std::begin(id);std::cout << * result << std::endl;// 以下,输出区间的终点与输入区间重叠,没问题std::copy(first,last,result);//(2 3 4 5 6) 5 6 7 8std::for_each(id.begin(),id.end(),display<int>{});std::cout << "\n";return 0;
    int ia[] = {0,1,2,3,4,5,6,7,8};//deque 拥有 RandomAcCGSSltGrCltorstd::deque<int>id(ia,ia+9);std::deque<int>::iterator first = id.begin();std::deque<int>::iterator last = id.end();++++first;  //advance(first,2);std::cout << *first << std::endl; //2----last;   //advance(last,-2);std::cout << *last << std::endl;  //7std::deque<int>::iterator result = std::begin(id);std::advance(result,4);std::cout << * result << std::endl;//以下,输出区间的起点与输入区间重叠,可能会有问题//本例结果错误,因为调用的 copy算法不再使用memmove ()执行实际复制操作std::copy(first,last,result);//0 1 2 3 (2 3 4 5 6std::for_each(id.begin(),id.end(),display<int>{});std::cout << "\n";return 0;
  • 请注意,如果你以vector取代上述的deque进行测试,复制结果将是正确的,因 为 vector迭代器其实是个原生指针(native pointer), 见4.2.3节,导致调用的copy算法以memmove ()执行实际复制操作
  • copy更改的是[result,result + (last-first)) 中的迭代器所指对象,而非更改迭代器本身。它会为输出区间内的元素赋予新值,而不是产生新的元素。它不能改变输出区间的迭代器个数。换句话说 , copy不能直接用来将元素插入空容器中。例子如下所示:如果不给myvector申请空间就会出错,只有申请空间之后初始化为0,然后执行覆盖原有数值的操作
  • 如果你想要将元素插入(而非赋值)序列之内,要么明白使用序列容器的insert 成员函数,要么使用 copy 算法并搭配 insert_iterator (8.3.1 节 )
  • 现在我们来看看 copy算法庞大的实现细节。下面是冰山一角,也是唯一的对外接口:
int main(int argc,char* argv[]) {int myints[]={10,20,30,40,50,60,70};std::vector<int>myvector(7);std::copy(myints,myints+7,myvector.begin());std::cout << "myvector contains:";for (std::vector<int>::iterator it = myvector.begin();it != myvector.end();++it) {std::cout << ' '<< *it;}std::cout << "\n";return 0;
}

  • 下面两个是多载函数,针对原生指针(可视为一种特殊的迭代器)const char*和 const wchar_t*, 进行内存直接拷贝操作:

  • 这里必须兵分两路来探讨。首先,_ copy_dispatch ()的完全泛化版根据迭代器种类的不同,调用了不同的 一copy()> 为的是不同种类的迭代器所使用的循 环条件不同,有快慢之别.

  • 这 是 __copy_dispatch ()完全泛化版的故事。现在回到前述兵分两路之处, 看看它的两个偏特化版本。这两个偏特化版是在“参数为原生指针形式”的前提下,希望进一步探测“指针所指之物是否具有trivial assignment operator (平凡 赋值操作符)” • 这一点对效率的影响不小,因为这里的复制操作是由assignment 操作符负责,如果指针所指对象拥有non-trivial assignment operator,复制操作就 一定得通过它来进行。但如果指针所指对象拥有的是trivial assignment operator, 复制操作可以不通过它,直接以最快速的内存对拷方式(memmove ())完成。C++语言本身无法让你侦测某个对象的型别是否具有trivial assignment operator,但是SGI STL采用所谓的  type_traits<> 编程技巧来弥补(见3.7节)。注意,通 过 “增 加一层间接性”的手法,我们便得以区分两个不同的  copy_t():

  • 第三个问题的解答是:我们以为vector的迭代器是random access iterator,没想到它事实上是个T* ;vector迭代器其实是原生指针。这就怪不得copy。 一旦面对vector 迭代器,就往T*的方向走去了;

6.4.4 copy_backw ard

  • 将[firstaast)区间内的每一个元素,以逆行的方向复制到 以 result-1为起点,方向亦为逆行的区间上。换句话说,copy_backward算法会
    执行赋值操作 * (result-1) = * (last-1) , * (result-2 ) = * (last-2 ) , 依此类推。
  • 返回一个迭代器:result- (last-first) □ copy_backward所接受的迭代器必
    须 是 Bidirectionallterators, 才 能 够 “倒行逆施”。你可以使用copy_backward算
    法,将任何容器的任何一段区间的内容,复制到任何容器的任何一段区间上°如果输
    入区间和输出区间完全没有重叠,当然毫无问题,否则便需特别注意

    int ia[] = {0,1,2,3,4,5,6,7,8};//输出区间的终点和输入区间重叠 没问题std::copy_backward(ia+2,ia+7,ia+9);std::for_each(ia,ia+9,display<int>{});//0 1 2 3 2 3 4 5 6std::cout<<std::endl;return 0;
    int ia[] = {0,1,2,3,4,5,6,7,8};//输出区间的起点和输入区间重叠 可能会有问题std::copy_backward(ia+2,ia+7,ia+5);std::for_each(ia,ia+9,display<int>{});//2 3 4 5 6 5 6 7 8//本例结果正确,因为调用的copy算法使用memmove ()执行实际复制操作std::cout<<std::endl;return 0;
template<class T>
struct display{void operator()(const T&x){std::cout << x << ' ';}
};
int main(int argc,char* argv[]) {int ia[] = {0,1,2,3,4,5,6,7,8};std::deque<int>id(ia,ia+9);std::deque<int>::iterator first = id.begin();std::deque<int>::iterator last = id.end();++++first; //advance(first,2);std::cout << *first << std::endl;----last;  //advance(last,-2);std::cout << *last << std::endl;std::deque<int>::iterator result = id.end();//输出区间的终点和输入区间重叠 没有问题std::copy_backward(first,last,result);std::for_each(id.begin(),id.end(),display<int>{});//0 1 2 3 2 3 4 5 6std::cout<<std::endl;return 0;
}
int main(int argc,char* argv[]) {int ia[] = {0,1,2,3,4,5,6,7,8};std::deque<int>id(ia,ia+9);std::deque<int>::iterator first = id.begin();std::deque<int>::iterator last = id.end();++++first; //advance(first,2);std::cout << *first << std::endl;----last;  //advance(last,-2);std::cout << *last << std::endl;std::deque<int>::iterator result = id.begin();std::advance(result,5);std::cout << *result << std::endl;//输出区间的起点和输入区间重叠 可能会有问题std::copy_backward(first,last,result);// 本例结果错误,因为调用的copy算法不再使用memmove()执行实际复制操作  但是我实际编码结果是正确的std::for_each(id.begin(),id.end(),display<int>{});//2 3 4 5 6 5 6 7 8std::cout<<std::endl;return 0;
}

STL源码剖析 基本算法 < stl_algobase.h >相关推荐

  1. STL源码剖析 基本算法 equal | fill | iter_awap | lexicographical_compare | max | min | swap |mismatch

    Equal 两个序列在[first,last)区间内相等,equal()返回true.以第一个序列作为基准,如果第二序列元素多不予考虑,如果要保证两个序列完全相等需要比较元素的个数 if(vec1.s ...

  2. STL源码剖析之算法:lower_bound

    lower_bound()函数返回能够插入value的第一个位置,该函数要求目标序列必须是有序的. template <class ForwardIterator, class T> in ...

  3. 【STL源码剖析】list模拟实现 | 适配器实现反向迭代器【超详细的底层算法解释】

    今天博主继续带来STL源码剖析专栏的第三篇博客了! 今天带来list的模拟实现! 话不多说,直接进入我们今天的内容! 前言 那么这里博主先安利一下一些干货满满的专栏啦! 手撕数据结构https://b ...

  4. STL源码剖析 数值算法 copy 算法

    copy复制操作,其操作通过使用assignment operator .针对使用trivial assignment operator的元素型别可以直接使用内存直接复制行为(使用C函数 memove ...

  5. STL源码剖析 算法开篇

    STL源码剖析 算法章节 算法总览_CHYabc123456hh的博客-CSDN博客 质变算法 质变算法 - 会改变操作对象的数值,比如互换.替换.填写.删除.排列组合.分隔.随机重排.排序等 #in ...

  6. STL源码剖析——P142关于list::sort函数

    在list容器中,由于容器自身组织数据的特殊性,所以list提供了自己的排序函数list::sort, 并且实现得相当巧妙,不过<STL源码剖析>的原文中,我有些许疑问,对于该排序算法,侯 ...

  7. STL源码剖析之配接器

    adapter(配接器)在STL组件的灵活组合运用上,扮演者转换器的角色.adapter来源于一种适配器模式,其功能是:将一个class接口转换为另一个class的接口,使得原本因接口不兼容而不能合作 ...

  8. STL(C++标准库,体系结构及其内核分析)(STL源码剖析)(更新完毕)

    文章目录 介绍 Level 0:使用C++标准库 0 STL六大部件 0.1 六大部件之间的关系 0.2 复杂度 0.3 容器是前闭后开(左闭右开)区间 1 容器的结构与分类 1.1 使用容器Arra ...

  9. 《STL源码剖析》学习-- 1.9-- 可能令你困惑的C++语法1

    最近在看侯捷的<STL源码剖析>,虽然感觉自己c++看得比较深一点,还是感觉还多东西不是那么明白,这里将一些细小的东西或者概念记录一下. 有些东西是根据<C++编程思想>理解的 ...

最新文章

  1. 如何判断车与路边线距离_6家快递公司共享分拣线配送车,效果如何?
  2. Day1-dns Extension
  3. wifi卡慢延迟高_健康生活好助手:华为智能体脂秤 WiFi 版 体验评测
  4. 黑马程序员python笔记_#华为云·寻找黑马程序员# 如何实现一个优雅的Python的Json序列化库...
  5. 安卓10侧边返回_安卓 10 细节曝光,这两个功能更好用了
  6. 【PostgreSQL-9.6.3】extract函数
  7. (十)洞悉linux下的Netfilteriptables:网络地址转换原理之SNAT
  8. 关于中标麒麟系统出现“网络管理器未响应”这件事的解决办法
  9. 一寸二寸证件大头半身照制作合成微信流量主小程序
  10. 两相四线步进电机C语言程序,两相四线步进电机时序
  11. Eclipse主题插件之Darkest Dark Theme with DevStyle
  12. Failed to process import candidates for configuration class :Annotation-specified bean name ‘XXX‘ fo
  13. html语言闪烁特效代码,css3 实现文字闪烁效果的三种方式示例代码
  14. 自我激励的有效方法20个(推荐)
  15. vw/vh:移动适配之vw/vh(使用方法)
  16. InnoDB和Myisam引擎的优缺点
  17. faiss search(检索)截断
  18. 一键生成smtp发件服务器文件,附录 D:创建 SMTP 服务器
  19. chart.js-一款轻巧实用的可视化图表工具
  20. 大学中明白的一百件事情

热门文章

  1. 华为笔记本软件商店_华为应用市场电脑版
  2. android设置多个按钮,android代码中设置两个按钮之间位置
  3. 【转】ABP源码分析五:ABP初始化全过程
  4. 计算机系统的备份与还原实验报告,数据库《数据备份与还原》实验报告.doc
  5. linux octave源码安装,在Linux操作系统上安装Octave的方法
  6. 【LightOJ - 1079】Just another Robbery(概率dp,概率背包)
  7. 【CodeForces - 523C】Name Quest (模拟)
  8. 【牛客 - 283E】贪心只能过样例(模拟)
  9. 【qduoj - 夏季学期创新题】最长公共子串(水题暴力枚举,不是LCS啊)
  10. 【CodeForces - 803D】Magazine Ad(二分答案)