为什么快排比堆排快

2024-01-13 02:38
文章标签 堆排 排比

本文主要是介绍为什么快排比堆排快,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今天作算法排序实验,发现相同的数据规模,快速排序比堆排序的效率高很多,并且随着数据规模的扩大,二者的差距不断扩大,快速排序的优势越来越明显。快速排序的时间复杂度近似线性增长,堆排序则要大很多。究其原因,应该有以下几个方面:
        在堆排序(小根堆)的时候,每次总是将最小的元素移除,然后将最后的元素放到堆顶,再让其自我调整。这样一来,有很多比较将是被浪费的,因为被拿到堆顶的那个元素几乎肯定是很大的,而靠近堆顶的元素又几乎肯定是很小的,最后一个元素能留在堆顶的可能性微乎其微,最后一个元素很有可能最终再被移动到底部。在堆排序里面有大量这种近乎无效的比较。随着数据规模的增长,比较的开销最差情况应该在(线性*对数)级别,如果数据量是原来的10倍,那么用于比较的时间开销可能是原来的10log10倍。
        堆排序的过程中,需要有效的随机存取。比较父节点和字节点的值大小的时候,虽然计算下标会很快完成,但是在大规模的数据中对数组指针寻址也需要一定的时间。而快速排序只需要将数组指针移动到相邻的区域即可。在堆排序中,会大量的随机存取数据;而在快速排序中,只会大量的顺序存取数据。随着数据规模的扩大,这方面的差距会明显增大。在这方面的时间开销来说,快速排序只会线性增长,而堆排序增加幅度很大,会远远大于线性。

        在快速排序中,每次数据移动都意味着该数据距离它正确的位置越来越近,而在堆排序中,类似将堆尾部的数据移到堆顶这样的操作只会使相应的数据远离它正确的位置,后续必然有一些操作再将其移动,即“做了好多无用功”。

        附:相关文章链接:

        调整法建堆过程的优化

        快速排序过程的优化

        数组规模对数组的读取效率的影响——顺序读取与随机读取

这篇关于为什么快排比堆排快的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/599973

相关文章

力扣爆刷第150天之TOP100五连刷(几数之和、堆排、合并链表)

力扣爆刷第150天之TOP100五连刷(几数之和、堆排、合并链表) 文章目录 力扣爆刷第150天之TOP100五连刷(几数之和、堆排、合并链表)一、15. 三数之和二、53. 最大子数组和三、912. 排序数组四、21. 合并两个有序链表五、1. 两数之和 一、15. 三数之和 题目链接:https://leetcode.cn/problems/3sum/descripti

排序算法 ---选择排序(直排,堆排)(java)

排序是在程序开发中常用的操作,也是个大公司面试的时候检验一个人编程能力的一个必考题,排序就是涉及到了算法了,今天下午就想着来搞一下排序的算法,算是对其有一个初步的了解吧,后天期中考了,挂科可就不是排序算法能够解决的问题了。 衡量一个算法优劣的标准: 1.时间复杂度,完成这个任务,算法所需要的时间。 2.空间复杂度,完成该任务,算法所需要的占用的内存空间,或者是所需要的外部辅助空间。 3.稳

排序——选择排序(直接选择排序和堆排)

本专栏和大家分享关于排序的算法,其中有插入排(直接插入排序和希尔排序)、选择排序(直接选择排序和堆排)、交换排序(冒泡排序和快速排序)、归并排序以及其他非基于比较的排序 本文与大家分享选择排序 目录  1.直接选择排序 优化: 时空复杂度: 稳定性: 2.堆排 向下调整的代码如下: 那么我们将数组向下调整后有什么用呢?? 具体代码: 时空复杂度: 稳定性: 感谢您的访

二分、快排、堆排与双指针

二分 int Binary_Search(vector<int> A,int key){int n=A.size();int low=0,high=n-1,mid;while(low<=high){mid=(low+high)/2;if(A[mid]==key)return mid;else if(A[mid]>key)high=mid-1;elselow=mid+1; }return -1;

堆排及时间复杂度分析

箴言: 初始阶段,不需要去纠结那一种更优美,非要找出那一种是最好的,其实能解决问题的就是好办法。 一,常见排序时间复杂度 冒泡快排归并堆排桶排时间O(n^2)O(nlogn)O(nlogn)O(nlogn)kn空间O(1)O(1)O(nlogn)O(1)kn 二,堆排 前情提要: 堆属于完全树,完全树可以理解为一个数组。如果不是完全树,就没办法和数组等价,就不会有下面这种父级和子级之间

堆排/快排/希尔/归并排序,谁更快?

文章目录 前因后果排序源码堆排代码快排代码希尔排序代码归并排序代码 测试环境100w数据量下的效率测试快排>希尔>归并>堆排 1000w数据下的测试快排>希尔>归并>堆排(差距拉大) 1亿数据量的测试快排>希尔>归并>堆排(差距拉到两倍) 总结皇城PK,个人的快排VS std::sort()数据量1亿测试测试时间对比(娱乐一下: 前因后果 好久没有再亲自写排序方

经典排序 之 堆排

1. 堆排的时间复杂度为O(nlogn), 空间复杂度为O(1)。 2. 先上原汁原味手写的代码 void heapAdjust(vector<int>& array, int len, int idx) {int left = idx * 2 + 1; //左孩子int right = left + 1; //右孩子if(left < len || right < len)

c语言经典算法—二分查找,冒泡,选择,插入,归并,快排,堆排

一、二分查找          1、前提条件:数据有序,随机访问;          2、实现:递归实现,非递归实现          3、注意事项:                 循环退出条件:low <=high,low = high.说明还有一个元素,该元素还要与key进行比较                 mid的取值:mid=(low + high)/2;mid = low

选择,插入,快排,堆排的Java实现

快速排序 选择基准元的方式: 对于分治算法,当每次划分时,算法若都能分成两个等长的子序列时,那么分治算法效率会达到最大。也就是说,基准的选择是很重要的。选择基准的方式决定了两个分割后两个子序列的长度,进而对整个算法的效率产生决定性影响。 最理想的方法是,选择的基准恰好能把待排序序列分成两个等长的子序列。 方法1 固定基准元 如果输入序列是随机的,处理时间是可以接受的。如果数组已经有序时,此时