【C++】STL中sort算法使用了什么排序算法?

参考:https://blog.csdn.net/u011386173/article/details/110394829

      STL所提供的各式各样的算法中,sort()是最复杂庞大的一个。这个算法接受两个RandomAccessIterators(随机存取迭代器),然后将区间内的所有元素以渐增方式由小到大重新排列。第二个版本则允许用户指定一个仿函数(functor),作为排序标准。STL的所有关系型容器(associative containers)都有用自动排序功能(底层结构采用RB-tree),所以不需要用到这个sort算法。至于序列式容器(sequence containers)中的stack、queue和priority-queue都有特别的出入口,不允许用户对元素排序。剩下vector、deque和list,前两者的迭代器属于RandomAccessIterators,适合使用sort算法,list的迭代器则属于BidirectioinalIterators,不在STL标准之列的slist,其迭代器更属于ForwardIterator,都不适合使用sort算法。如果要对list或slist排序,应该使用它们自己提供的member functions sort()。STL的sort算法, 数据量大时采用Quick Sort,分段递归排序。一旦分段后的数据量小于某个门槛,为避免Quick Sort的递归调用带来过大的额外负荷(overhead),就改用 Insertion Sort。如果递归层次过深,还回改用 Heap Sort。                                      —— From《STL源码剖析》      

源码:

stl_algo.h:

template <class _RandomAccessIter>
inline void sort(_RandomAccessIter __first, _RandomAccessIter __last) {__STL_REQUIRES(_RandomAccessIter, _Mutable_RandomAccessIterator);__STL_REQUIRES(typename iterator_traits<_RandomAccessIter>::value_type,_LessThanComparable);if (__first != __last) {__introsort_loop(__first, __last,__VALUE_TYPE(__first),__lg(__last - __first) * 2); //快速排序__final_insertion_sort(__first, __last); //插入排序}
}
template <class _RandomAccessIter, class _Tp, class _Size>
void __introsort_loop(_RandomAccessIter __first,_RandomAccessIter __last, _Tp*,_Size __depth_limit)
{while (__last - __first > __stl_threshold) {//1、__depth_limit 是控制递归深度if (__depth_limit == 0) {partial_sort(__first, __last, __last); //递归深度过深时采用堆排序return;}--__depth_limit;//2、单边递归_RandomAccessIter __cut =__unguarded_partition(__first, __last,_Tp(__median(*__first,*(__first + (__last - __first)/2),*(__last - 1)))); //3、__median三点取中法__introsort_loop(__cut, __last, (_Tp*) 0, __depth_limit);__last = __cut;}
}
template <class _RandomAccessIter>
inline void partial_sort(_RandomAccessIter __first,_RandomAccessIter __middle,_RandomAccessIter __last) {__STL_REQUIRES(_RandomAccessIter, _Mutable_RandomAccessIterator);__STL_REQUIRES(typename iterator_traits<_RandomAccessIter>::value_type,_LessThanComparable);__partial_sort(__first, __middle, __last, __VALUE_TYPE(__first));
}template <class _RandomAccessIter, class _Tp>
void __partial_sort(_RandomAccessIter __first, _RandomAccessIter __middle,_RandomAccessIter __last, _Tp*) {make_heap(__first, __middle); //线性建堆for (_RandomAccessIter __i = __middle; __i < __last; ++__i) //堆排序if (*__i < *__first) __pop_heap(__first, __middle, __i, _Tp(*__i),__DISTANCE_TYPE(__first));sort_heap(__first, __middle); //堆排序
}
template <class _RandomAccessIterator>
inline void 
make_heap(_RandomAccessIterator __first, _RandomAccessIterator __last)
{__STL_REQUIRES(_RandomAccessIterator, _Mutable_RandomAccessIterator);__STL_REQUIRES(typename iterator_traits<_RandomAccessIterator>::value_type,_LessThanComparable);__make_heap(__first, __last,__VALUE_TYPE(__first), __DISTANCE_TYPE(__first));
}template <class _RandomAccessIterator, class _Tp, class _Distance>
void 
__make_heap(_RandomAccessIterator __first,_RandomAccessIterator __last, _Tp*, _Distance*)
{if (__last - __first < 2) return;_Distance __len = __last - __first;_Distance __parent = (__len - 2)/2;while (true) { //线性建堆__adjust_heap(__first, __parent, __len, _Tp(*(__first + __parent)));if (__parent == 0) return;__parent--; //__parent就是线性建堆时逆序扫描}
}

总结来说,STL 中的排序算法sort有3个步骤:

控制递归深度 --> 使用了堆排序 --> 线性建堆

单边递归

三点取中

排序区间大小:当排序区间大的时候使用了快速排序,当排序区间过小的时候停止快速排序,使用插入排序。也就是说经过快速排序后,整段空间被分割为了几段,每段里面的数不是有序的,但是第一段的所有数都小于第二段,第二段的所有数都小于第三段…,最后对每一段插入排序使其成为有序序列。

也就是说sort算法将三种排序进行了融合:

对于区间较大的情况,使用了快速排序;
当递归深度过深时,使用堆排序;
最终的排序整理,使用插入排序。
即STL中的sort算法是快排、插入排序和堆排序的综合

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/777958.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Dubbo的服务注册与发现原理、Java如何实现Dubbo的服务注册与发现

1、Dubbo的服务注册与发现原理 Dubbo的服务注册与发现原理主要基于Zookeeper实现。 1、服务提供者在启动时&#xff0c;会将自己的服务地址和相关信息注册到Zookeeper中的指定节点。注册的信息包括服务名称、服务地址、服务版本号等。 2、服务消费者在启动时&#xff0c;会连…

【GitLab】Ubuntu使用宝塔安装GitLab最新社区版

首先在Ubuntu安装宝塔面板 在官网可以找到脚本一键安装 安装GitLab社区版 然后在宝塔面板的“软件商店”里面找到GitLab最新社区版 12.8.1一键安装 安装过程中可能出现以下问题&#xff1a; 1.卡在ruby_block[wait for logrotate service socket] action run 解决办法&…

离心式风机运行效率测算

1.总压静压动压&#xff1b; 2.动压0.5空气体密度风速2&#xff1b; 风机所需功率P&#xff08;KW&#xff09;&#xff1a;PQp/&#xff08;36001000η0η1&#xff09; Q—风量&#xff0c;m3/h&#xff1b; p—风机的全风压&#xff0c;Pa&#xff1b; η0—风机的内效率&a…

iOS library not found for -lMBProgressHUD

0x00 前因 一开始是使用 CocoaPods 管理 MBProgressHUD&#xff0c;后来直接导入 MBProgressHUD 源码&#xff0c;就出现了这个错误&#xff1a;library not found for -lMBProgressHUD 0x01 后果 在 Xcode 工程目录中找到文件夹&#xff1a;Frameworks 看看里面是否有个红色…

计算机网络-RIP动态路由协议简介

一、概述 前面我们学习了动态路由协议按照工作机制及算法划分可以分为&#xff1a;距离矢量路由协议DV型和链路状态路由协议LS型。RIP就是典型的距离矢量路由协议&#xff0c;但是实际工作中用得已经比较少了。 距离矢量路由协议DV: RIP 链路状态路由协议LS: OSPF IS-IS 二、RI…

提升 Vim 技能的四种方法

如果您经常使用安全shell&#xff1a; $ ssh userhostname.provider.com 并使用虚拟专用服务器 (VPS) 或本地虚拟化容器&#xff0c;就此而言&#xff0c;您可以从强大的 Vim 技能中受益匪浅。 一、将 Vim 设置为默认文本编辑器 1、将 Vim 设置为 Bash 中的默认值 Bash 通…

深入Spark与LDA:大规模文本主题分析实战

使用LDA模型和Spark进行文本主题分析 本篇博客介绍了如何使用LDA&#xff08;潜在狄利克雷分配&#xff09;模型和Spark进行文本主题分析。我们的目标是从大量的用户评论中提取出主题。 1. 环境设置 首先&#xff0c;我们需要导入所需的库&#xff0c;包括jieba&#xff08;…

Java八股文(JVM)

Java八股文のJVM JVM JVM 什么是Java虚拟机&#xff08;JVM&#xff09;&#xff1f; Java虚拟机是一个运行Java字节码的虚拟机。 它负责将Java程序翻译成机器代码并执行。 JVM的主要组成部分是什么&#xff1f; JVM包括以下组件&#xff1a; ● 类加载器&#xff08;ClassLoa…

当当狸智能激光雕刻机 多种材质自由雕刻,轻松打造独一无二的作品

提及“激光雕刻”&#xff0c;大多数人的印象一般都是&#xff1a;笨重巨大、价格昂贵、操作复杂、使用门槛较高、调试难度大...不是普通人能够随意操作的&#xff0c;让人望尘莫及。 而小米有品上新的这台「当当狸桌面智能激光雕刻机L1」&#xff0c;将超乎你的想象&#xff…

JimuReport积木报表 v1.7.4 公测版本发布,免费的JAVA报表工具

项目介绍 一款免费的数据可视化报表&#xff0c;含报表和大屏设计&#xff0c;像搭建积木一样在线设计报表&#xff01;功能涵盖&#xff0c;数据报表、打印设计、图表报表、大屏设计等&#xff01; Web 版报表设计器&#xff0c;类似于excel操作风格&#xff0c;通过拖拽完成报…

【C语言基础】:数据在内存中的存储

文章目录 一、整数在内存中的存储二、大小端字节序和字节序判断1. 为什么有大小端&#xff1f;2. 练习 三、浮点数在内存中的存储1. 浮点数的存储1.1 浮点数的存储过程1.2 浮点数取的过程 四、题目解析 书山有路勤为径&#xff0c;学海无涯苦作舟。 创作不易&#xff0c;宝子们…

google浏览器下载文件提示无法安全地下载怎么解决?

在使用google浏览器下载文件的时候,弹出了“无法安全下载”的提示,搞了文件都下载不下来,网上查了一下,是因为chrome认为使用非https链接下载文件是不安全的,在新版本中阻止了用户下载。 目录 1、打开google浏览器的设置

轻量级检测模型效果一定差?基于轻量级目标检测模型构建布匹瑕疵检测模型,对比分析不同分辨率图像尺度对模型效果的影响

瑕疵类的检测模型如&#xff1a;工业部件瑕疵、瓷砖瑕疵、PCB瑕疵、布匹瑕疵等等&#xff0c;在我们之前的博文中已经有过很多相关的开发实践了&#xff0c;这里就不再一一列举了&#xff0c;感兴趣的话可以直接搜索关键字信息博文内容即可一键直达。 因为本文的实验对象选择的…

structured bindings is supported from c++17

结构化绑定可以绑定结构体、数组和 tuple-like 对象。 完整示例&#xff1a; #include <iostream> #include <format> #include <iomanip>void test_00(){struct Box{int width_;int height_;std::string name_;};Box box{3,4,"amazing"};auto [w…

IT服务营销管理案例分析题

习题一 企业随着业务的蓬勃发展&#xff0c;所投入的基础设施资源不断增加。企业员工数倍数增长&#xff0c;办公场地、办公环境等要求也越来越高。 可是该企业的IT部门人员短缺&#xff0c;对IT管理还处于被动的“救火”阶段&#xff0c;每天至少15个突发故障&#xff0c;故障…

YOLOv9改进策略 | 损失函数篇 | InnerIoU、InnerMPDIoU二次创新损失函数助力小目标检测(全网独家首发)

一、本文介绍 本文给大家带来的改进机制是InnerIoU以及包含其它二次创新的代码InnerCIoU、InnerMPDIoU等一些列利用Inner形成的二次创新损失函数&#xff0c;到此大家可能比较模糊为啥Inner能够和其他损失函数形成二次创新&#xff0c;Inner又是一个什么样的机制&#xff0c;开…

【1】网络协议基础概念

【1】网络协议基础知识 1、互联网2、为什么要学习网络协议3、学习中需要搭建的环境4、客户端-服务器5、Java 的跨平台原理6、C/C的跨平台原理7、一个简单的SpringBoot项目(1) pom.xml(2) application.yml(3) NetworkStudyApp.java(4) SwaggerConfig.java(5) HelloWorldControll…

Git--08--Git分支合并操作

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 Git分支合并操作案例流程客户端&#xff1a;GitExtensions操作步骤&#xff1a;A操作步骤&#xff1a;B操作步骤&#xff1a;C操作步骤&#xff1a;D操作步骤&#…

Vue挂载全局方法

简介&#xff1a;有时候&#xff0c;频繁调用的函数&#xff0c;我们需要把它挂载在全局的vue原型上&#xff0c;方便调用&#xff0c;具体怎么操作&#xff0c;这里来记录一下。 一、这里以本地存储的方法为例 var localStorage window.localStorage; const db {/** * 更新…

LockSupport与线程中断机制

中断机制是个协商机制 Interrupt(): 将中断状态设置为true Interrupted():&#xff08;静态方法&#xff09; 1.返回当前线程的中断状态 2.将中断状态清零并设置为false is Interrupted(): 判断当前线程是否被中断 如何停止中断运行中的线程&#xff1f; 一个线程不应该由…