容器算法

<algorithm>c++自带的容器算法,提供一系列实用的算法。在谈到容器算法,我们大概率会用到谓词predicate,谓词返回的类型是布尔类型(bool)可以是lambda表达式、函数对象以及其它可调用的对象。

查找

  1. find()查找元素

    find接受三个参数,第三个参数是值类型,setmap自带count函数也能实现这样的功能,返回值0表示不存在。为了方便本次连带find_iffind_if_notfind_first_offind_endadjacent_find一起举例。

    1. vector<int> a{1, 1, 2, 3, 4, 4, 5};
    2. vector<int> b{5, 6, 6};
    3. find(a.begin(), a.end(), 3); // 返回的是迭代器,未查找到返回a.end()
    4. find_if(a.begin(), a.end(), [](auto x){return x != 3;}); //接受一个一元谓词,找到第一个满足条件的元素
    5. find_if_not(a.begin(), a.end(), [](auto x){return x != 3;}); //接受一个一元谓词,找到第一个不满足条件的元素
    6. // find_first_of找到存在于第一个范围的第二个范围中的第一个元素(可能并不是第二个范围第一个元素),返回迭代器,也支持一个二元谓词。
    7. vector<int> a{1, 1, 2, 3, 4, 4, 5};
    8. vector<int> b{6, 5, 6};
    9. auto it = find_first_of(a.begin(), a.end(), b.begin(), b.end());
    10. cout << *it << endl; // 返回5
    11. // find_end找到最后一个匹配子序列的位置,返回最后一个子序列开始的迭代器,也支持一个二元谓词
    12. vector<int> a{1, 1, 2, 3, 1, 4, 1, 2};
    13. vector<int> b{1, 2};
    14. auto it = find_end(a.begin(), a.end(), b.begin(), b.end());
    15. cout << distance(a.begin(), it) << endl; // 6指向第7个元素
    16. // adjcent_find找到两个值邻近的元素,返回指向找到的第一个元素的迭代器,接受一个二元谓词
    17. vector<int> a{1, 1, 2, 3, 4, 1, 2};
    18. vector<int> b{1, 2};
    19. auto it = adjacent_find(a.begin(), a.end());
    20. auto it = adjacent_find(a.begin(), a.end(),
    21. [](auto a, auto b)->bool{return a + b = 10;});
    22. cout << *it << endl;

去重

  1. unique(nums.begin(), nums.end())除掉连续相同的值

    unique函数的作用是删除掉的连续的相同的值,unique还支持传入二元谓词(可以理解为一个参数的函数),会返回一个迭代器it,但是并不是end(),中间是未指定的值(访问可能会产生未定义行为)。注意不要用set,返回的都是常量迭代器是无法改变的,其指向的元素无法改变。

    1. // 无谓词写法
    2. vector<int> a{1, 1, 2, 3, 4, 4, 5};
    3. auto it = unique(a.begin(), a.end()); // 此时a为{1, 2, 3, 4, 5}
    4. for_each(a.begin(), a.end(), [](auto it){cout << it << endl;}); // 输出为{1, 2, 3, 4, 5, 4, 5}
    5. for_each(a.begin(), it, [](auto it){cout << it << endl;}); // 输出为{1, 2, 3, 4, 5}
    6. // 有二元谓词
    7. vector<int> a{1, 1, 2, 3, 4, 4, 5};
    8. auto it = unique(a.begin(),
    9. a.end(), [](auto a, auto b) -> bool{return a != b;}); // 此时a为{1, 2, 3, 4, 5},lambda表达式作为一个二元谓词,每次从数组中取两个严肃进行判断,然后删除不相等的元素。
    10. for_each(a.begin(), a.end(), [](auto it) -> int{cout << it << endl;}); // Error,这是因为unique将返回的迭代器到之后的end()迭代器指向的值删除了,但是空间还在。此时访问,会发生未定义行为。
    11. for_each(a.begin(), it, [](auto xs){cout << xs << endl;}); // 1,1
    12. cout << distance(a.begin(), a.end()) << endl; // 7,distance求算距离是7
  2. unique_copy()除掉连续相同的值并复制到目标容器

    注意,unique_copy并不进行内存分配,只是赋值给另一个容器,如果访问未分配内存的容器,那么会产生未定义行为。同样,也支持二元谓词,通过二元谓词判定移除连续值。

    1. vector<int> a{1, 1, 2, 3, 4, 4, 5};
    2. vector<int> b(7, 0);
    3. auto it = unique_copy(a.begin(), a.end(), b.begin());
    4. for_each(b.begin(), it, [](auto xs){cout << xs << endl;}); // out: 1 2 3 4 5
    5. // 二元谓词, 一般不要使用二元谓词与数字直接进行比较,可以使用remove实现
    6. auto it = unique_copy(a.begin(),
    7. a.end(), b.begin(), [x](auto a, auto b)
    8. -> bool{return a == 3;});
    9. for_each(b.begin(), it, [](auto xs){cout << xs << endl;}); // out: 1 1 2 3,解释:值等于3才会进行unique_copy
    10. auto it = unique_copy(a.begin(),
    11. a.end(), b.begin(), [x](auto a, auto b)
    12. -> bool{return a == b;});
    13. for_each(b.begin(), it, [](auto xs){cout << xs << endl;}); // out: 1 2 3 4 5,解释:值等于3才会进行unique_copy

排序

  1. sort()对容器进行排序

    sort用于实现容器元素的排序,sort也同样接受二元谓词

    1. sort(a.begin(), a.end());
    2. sort(a.begin(), a.end(), greater<>()); // 使用库中自带的greater\less对象

迭代器差值

  1. distance()求迭代之间距离

    distance用于求算两个迭代器之间的差值,只用于同一容器, setmapvector都可以用。

    1. distance(c.begin(), c.end());

遍历容器

  1. for_each()for_each_n遍历容器每个元素并执行函数规定的操作,第三个参数是一个函数。如果要对迭代器指向的元素修改,那么set就是不可以的,因为set返回的是常量迭代器。

    1. array<int, 5> a{1, 2, 3, 4, 5};
    2. for_each(a.begin(), a.end(), [](int& x){cout<< x; x *= 2;}); // 此时元素值会发生改变,因为捕获的是引用。
    3. for_each(a.begin(), a.end(), [](auto x) -> void{x *= 2; cout << x;}); // 此时元素值不会发生改变,捕获的是值,会变为副本。
    4. for_each_n(a.begin, 5, [](auto x) -> void{x *= 2; cout << x;}); // 第二个参数指定操作的数量n, 对n个元素进行操作。

复制元素

  1. copy()能够将容器中的元素复制到输出迭代器之后out iterator

    1. vector<vector<int>> sx(1);
    2. copy(sx.begin(), sx.end() , sx.begin()); // 可以将[begin, end)之间的元素复制到从begin()开始的位置。
    3. // 一个有意思的做法是使用back_inserter指明要进行尾插法,某些支持双向插入的容器也可以转变为尾插方法,省去了一些需要选择插入函数的麻烦。
    4. // set、map类红黑树容器不存在后插方法,无法进行此类调用,注意
    5. copy(sx.begin(), sx.end() , back_inserter(sx));
    6. vector<int> a{1, 2, 3};
    7. set<int> s;
    8. copy(a.begin(), a.end() , back_inserter(s)); // Error
    9. // 接下来我们考虑一个向vector<vector<int>>复制的问题
    10. vector<vector<int>> v(1); // 此时外部vector容量为1,size也是1。说明内部存在一个vector容器元素,但是此时vector[0]的size和capacity都是0,是一个空容器。
    11. copy(v.begin(), v.end(), back_inserter(v)); // 此时内部空容器是2个
  2. copy_backward()能够将容器中元素复制到输出迭代器之前。

    map、set类也不支持。这个容器算法会覆盖掉输出迭代器之前的元素

    1. vector<int> a{1, 2, 3, 4, 5, 6};
    2. set<int> s;
    3. copy_backward(a.begin(), a.end() , s); // Error
    4. copy_backward(a.begin(), a.begin() + 3, a.end()); // OK a{1, 2, 3, 1, 2, 3}
    5. copy_backward(a.begin(), a.begin() + 3, a.begin()); // OK,但是数组不会发生任何改变,相当于复制到begin()迭代器之前没有意义。

C++容器算法的更多相关文章

  1. 【足迹C++primer】35、特定容器算法

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/cutter_point/article/details/33732681 特定容器算法 lst.me ...

  2. STL的容器算法迭代器的设计理念

    1) STL的容器通过类模板技术,实现数据类型和容器模型的分离. 2) STL的迭代器技术实现了遍历容器的统一方法:也为STL的算法提供了统一性. 3) STL的函数对象实现了自定义数据类型的算法运算 ...

  3. c++容器 算法 迭代

    #include <iostream> #include <vector> using namespace std; int main() { // 创建一个向量存储 int ...

  4. C++容器和算法

    转自:http://www.cnblogs.com/haiyupeter/archive/2012/07/29/2613145.html 容器:某一类型数据的集合. C++标准顺序容器包括:vecto ...

  5. 4_STL设计理念_算法

    STL算法,容器,迭代器的设计理念1.STL容器通过 类模板 技术,实现 数据类型 和 容器模型的分离:2.迭代器技术 实现了 遍历和操作容器的统一方法3.STL算法设计理念:通过预定义的函数对象和函 ...

  6. C++ Primer :第十章 :泛型算法之再探迭代器以及其他算法

    除了为每个容器定义的迭代器之外,标准库在头文件<iterator>还定义了额外集中迭代器, 包括: 插入迭代器,这些迭代器被绑定到一个容器上,可以向容器插入元素. 流迭代器,    这些迭 ...

  7. C++ Primer : 第十章 : 泛型算法 之 只读、写和排序算法

    大多数算法都定义在<algorithm>头文件里,而标准库还在头文件<numeric>里定义了一组数值泛型算法,比如accumulate. ●  find算法,算法接受一对迭代 ...

  8. C++标准库之泛型算法

    本文中算法都是指泛型算法. 基本要点: 1)算法使用迭代器进行操作. 2)不依赖容器,但容器希望使用算法,就必须提供接口. 3)通用算法永远不会执行容器操作.操作仅指:更改容器大小的操作.但,容器内部 ...

  9. C++ STL常用容器浅析

    首先要理解什么是容器,在C++中容器被定义为:在数据存储上,有一种对象类型,它可以持有其它对象或指向其它对象的指针,这种对象类型就叫做容器.简单来说 容器就是包含其他类的对象们的对象,当然这种(容器) ...

  10. c++常用小算法

    这篇文章列出了一些简单常用的C++容器算法,C++标准库中事实上提供了很多的算法并且有详细的介绍.如果需要详细的了解这些算法可以 参考C++在线参考手册 algrithm . 1 排序 在 #incl ...

随机推荐

  1. Linux 应用案例开发手册——基于Zynq-7010/20工业开发板

    目 录 1 开发案例说明 4 2 Linux 常用开发案例 4 2.1 tl_led_flash 案例 4 2.2 tl_key_test 案例 7 2.3 tl_can_echo 案例 11 2.4 ...

  2. scala偏函数小栗子

    package cn.beicaiqm.scala.day04 /** * Created by Administrator on 2018/6/1. * 被包在花括号内没有match的一组case语 ...

  3. 不是人家太装逼,而是我们太low

    在一个社团的迎新的时候,每个人自我介绍.等到一个一身LV,爱马仕的女孩子自我介绍,说起爱好,她想了想说:喜欢跑车.然后很淡定的坐下了.很多同学你看我我看你,投以"炫富"的判断目光- ...

  4. 典型性相关分析在SPSS中的实现

    典型性相关分析是研究两组变量(每组变量中都可能有多个指标)之间相关关系的一种多元统计方法.它能够揭示出两组变量之间的内在联系. 本文着重模型在spss中的应用,通过一道例题解释各个指标的意义.详细推导 ...

  5. springboot 整合 pagehelper

    pom.xml <dependency> <groupId>com.github.pagehelper</groupId> <artifactId>pa ...

  6. Java 知识总结大汇总!看完哪个都变大佬!

    免费编程资源大全项目:https://github.com/liyupi/free-programming-resources 大家好,我是鱼皮,今天分享 十几个 让人直呼 "哇塞" ...

  7. Python pluggy框架基础用法总结

    代码为例进行说明 实践环境 Python 3.6.5 pluggy 0.13.0 例1 注册类函数为插件函数 #!/usr/bin/env python # -*- coding:utf-8 -*- ...

  8. 机器学习:详解多任务学习(Multi-task learning)

    详解多任务学习 在迁移学习中,步骤是串行的,从任务\(A\)里学习只是然后迁移到任务\(B\).在多任务学习中,是同时开始学习的,试图让单个神经网络同时做几件事情,然后希望这里每个任务都能帮到其他所有 ...

  9. 新年切红包-scratch小游戏

    程序说明: <新年切红包>是一款Scratch制作的小游戏,灵感来源于流行的切水果游戏.在这个游戏中,玩家需要用鼠标切割屏幕上不断飞出的红包,切割到红包将获得金币奖励,而切割到爆竹则会导致 ...

  10. sharding-jdbc 兼容 MybatisPlus的动态数据源

    背景:之前的项目做读写分离的时候用的 MybatisPlus的动态数据做的,很多地方使用的@DS直接指定的读库或者写库实现的业务:随着表数据量越来越大,现在打算把比较大的表进行水平拆分,准备使用 Sh ...