• C++11标准模板(STL)- 算法(std::stable_sort)


    定义于头文件 

    算法库提供大量用途的函数(例如查找、排序、计数、操作),它们在元素范围上操作。注意范围定义为 [first, last) ,其中 last 指代要查询或修改的最后元素的后一个元素。

    将范围内的元素排序,同时保持相等的元素之间的顺序

    std::stable_sort

    template< class RandomIt >
    void stable_sort( RandomIt first, RandomIt last );

    (1)

    template< class ExecutionPolicy, class RandomIt >
    void stable_sort( ExecutionPolicy&& policy, RandomIt first, RandomIt last );

    (2)(C++17 起)

    template< class RandomIt, class Compare >
    void stable_sort( RandomIt first, RandomIt last, Compare comp );

    (3)

    template< class ExecutionPolicy, class RandomIt, class Compare >
    void stable_sort( ExecutionPolicy&& policy, RandomIt first, RandomIt last, Compare comp );

    (4)(C++17 起)

    以升序排序范围 [first, last) 中的元素。保证保持等价元素的顺序。

    1) 用 operator< 比较元素。

    3) 用给定的比较函数 comp 比较元素。

    2,4) 同 (1,3) ,但按照 policy 执行。这些重载不参与重载决议,除非 std::is_execution_policy_v> 为 true 。

    参数

    first, last-要排序的元素范围
    policy-所用的执行策略。细节见执行策略。
    comp-比较函数对象(即满足比较 (Compare) 概念的对象),若第一参数小于(即序于)第二参数则返回 ​true 。

    比较函数的签名应等价于如下:

     bool cmp(const Type1 &a, const Type2 &b);

    虽然签名不必有 const & ,函数也不能修改传递给它的对象,而且必须接受(可为 const 的)类型 Type1Type2 的值,无关乎值类别(从而不允许 Type1 & ,亦不允许 Type1 ,除非 Type1 的移动等价于复制 (C++11 起))。
    类型 Type1 与 Type2 必须使得 RandomIt 类型的对象能在解引用后隐式转换到这两个类型。 ​

    类型要求
    - RandomIt 必须满足值可交换 (ValueSwappable) 和 遗留随机访问迭代器 (LegacyRandomAccessIterator) 的要求。
    - 解引用 RandomIt 结果的类型必须满足可移动赋值 (MoveAssignable) 和可移动构造 (MoveConstructible) 的要求。

    返回值

    (无)

    复杂度

    应用 cmp O(N·log(N)2) 次,其中 N = std::distance(first, last) 。若额外内存可用,则复杂度为 O(N·log(N)) 。

    异常

    拥有名为 ExecutionPolicy 的模板形参的重载按下列方式报告错误:

    • 若作为算法一部分调用的函数的执行抛出异常,且 ExecutionPolicy 为标准策略之一,则调用 std::terminate 。对于任何其他 ExecutionPolicy ,行为是实现定义的。
    • 若算法无法分配内存,则抛出 std::bad_alloc 。

    注意

    此函数试图分配等于待排序序列长度的临时缓冲区。若分配失败,则选择较低效的算法。

    调用示例

    1. #include <iostream>
    2. #include <algorithm>
    3. #include <functional>
    4. #include <vector>
    5. #include <iterator>
    6. #include <time.h>
    7. using namespace std;
    8. struct Cell
    9. {
    10. int x;
    11. int y;
    12. Cell &operator +=(const Cell &cell)
    13. {
    14. x += cell.x;
    15. y += cell.y;
    16. return *this;
    17. }
    18. bool operator <(const Cell &cell) const
    19. {
    20. return x < cell.x;
    21. }
    22. };
    23. std::ostream &operator<<(std::ostream &os, const Cell &cell)
    24. {
    25. os << "{" << cell.x << "," << cell.y << "}";
    26. return os;
    27. }
    28. int main()
    29. {
    30. srand((unsigned)time(NULL));;
    31. std::cout.setf(std::ios_base::boolalpha);
    32. auto func1 = []()
    33. {
    34. Cell cell{std::rand() % 10 + 100, std::rand() % 10 + 100};
    35. return cell;
    36. };
    37. vector<Cell> cells(8);
    38. std::generate(cells.begin(), cells.end(), func1);
    39. std::cout << "cells : ";
    40. std::copy(cells.begin(), cells.end(), std::ostream_iterator<Cell>(std::cout, " "));
    41. std::cout << std::endl;
    42. std::cout << "is_sorted: " << std::is_sorted(cells.begin(), cells.end());
    43. std::cout << std::endl << std::endl;
    44. std::stable_sort(cells.begin(), cells.end());
    45. std::cout << "cells : ";
    46. std::copy(cells.begin(), cells.end(), std::ostream_iterator<Cell>(std::cout, " "));
    47. std::cout << std::endl;
    48. std::cout << "is_sorted: " << std::is_sorted(cells.begin(), cells.end());
    49. std::cout << std::endl << std::endl;
    50. auto is_sortf = [](const Cell & a, const Cell & b)
    51. {
    52. if (a.x == b.x)
    53. {
    54. return a.y > b.y;
    55. }
    56. return a.x > b.x;
    57. };
    58. std::generate(cells.begin(), cells.end(), func1);
    59. std::cout << "cells : ";
    60. std::copy(cells.begin(), cells.end(), std::ostream_iterator<Cell>(std::cout, " "));
    61. std::cout << std::endl;
    62. std::cout << "is_sorted: " << std::is_sorted(cells.begin(), cells.end(), is_sortf);
    63. std::cout << std::endl << std::endl;
    64. std::stable_sort(cells.begin(), cells.end(), is_sortf);
    65. std::cout << "cells : ";
    66. std::copy(cells.begin(), cells.end(), std::ostream_iterator<Cell>(std::cout, " "));
    67. std::cout << std::endl;
    68. std::cout << "is_sorted: " << std::is_sorted(cells.begin(), cells.end(), is_sortf);
    69. std::cout << std::endl << std::endl;
    70. return 0;
    71. }

    输出

  • 相关阅读:
    抓包工具fiddler
    Java开发面试常见问题总结
    Rt-Thread 移植2--线程定义与切换(KF32)
    【图论】树的重心
    【C++初阶】类和对象(二)
    以分割栅格为例实现FME模板的方案优化
    java之面向对象编程特性:继承
    希尔排序算法(代码实现) [数据结构][Java]
    jar包或exe程序设置为windows服务
    研发中台拆分过程的一些心得总结
  • 原文地址:https://blog.csdn.net/qq_40788199/article/details/127921819