听说你要排上亿个数据之 HeapSort ?

前言 : 来来来,今天我们来说说一个用来排大量数据所用的基础比较排序吧~

注:阅读本文学习新技能的前置要求为:了解什么是二叉树及其数组性质,如果未达到要求的同学请直接看完图默默点右上角的×就好=- =~

在那之前我们先来看一个图

嗯,是不是很好玩?这玩意就是所谓的 HeapSort (堆排序)

而我们今天就来说说怎么像图例所示那样来进行我们的排序操作吧~

所谓的 HeapSort 呢,就是在一直做一件事,这件事用高雅点的名词就叫: HeapAdjust (调整最大堆)

在说明最大堆之前先提问,什么是堆呢?

沙堆,土堆,石堆,这些事物的特点是什么呢?

就好比金字塔,内容物是从底下到顶逐渐减少的.而我所说的最大堆,就是在说这样的一种堆

并且其最大堆的内容物(沙土石头或称 Key )呈现出其堆顶的价值(数值)最大,然后向下逐级递减.

那么,怎么产生这样的最大堆呢?

下面我贴一段来自于百度百科的代码吧~

百度百科 - 堆排序

// array 是待调整的堆数组, i 是待调整的数组元素的位置, length 是数组的长度
// 本函数功能是:根据数组 array 构建大根堆
void HeapAdjust(int array[], int i, int length)
{
for(int Child, Temp; 2 * i + 1 < length; i = Child)
{
//子结点的位置 = 2 *(父结点位置)+ 1
Child = 2 * i + 1;
//得到子结点中较大的结点
if(Child + 1 < length && array[Child + 1] > array[Child]) ++Child;
//如果较大的子结点大于父结点那么把较大的子结点往上移动,替换它的父结点
if(array[i] < array[Child]) Temp = array[i], array[i] = array[Child], array[Child] = Temp;
//否则退出循环
else break;
}
}
//堆排序算法
void HeapSort(int array[], int length)
{
int i;
// 调整序列的前半部分元素,调整完之后第一个元素是序列的最大的元素
// length / 2 - 1 是最后一个非叶节点,此处"/"为整除
for(i = length / 2 - 1; i >= 0; --i) HeapAdjust(array, i, length);
// 从最后一个元素开始对序列进行调整,不断的缩小调整的范围直到第一个元素
for(i = length - 1; i > 0; --i)
{
// 把第一个元素和当前的最后一个元素交换,保证当前的最后一个位置的元素都是在现在的这个序列之中最大的
array[i] ^= array[0], array[0] ^= array[i], array[i] ^= array[0];
// 不断缩小调整heap的范围,每一次调整完毕保证第一个元素是当前序列的最大值
HeapAdjust(array, 0, i);
}
}

所有要做的工作就在这里面,看上去是挺精简的了我觉得=- =.

那我们来说说它是怎么工作的吧~

首先 HeapSort 这个函数要做第一次预处理工作,这个工作的目的就是要把这个数组调整为最大堆

即倒序执行 HeapAdjust

但是并不是倒数第一个开始执行,而是倒数第二层中存在子节点的树结点开始(length / 2 - 1)。

这是什么意思呢?

这是因为HeapAdjust操作原理为:

首先选出其左右结点中较大的结点,然后将其与其父结点相比较,若大于则Swap之(此时构建最大堆).

然后对当前子结点执行上面的重复操作.

好的,当倒序到0的时候,就成功将这个最大堆建立起来了,现在需要做什么呢?

现在已知的条件是,顶端为这个堆的最大值,那么,这个最大值与数组最后一个元素交换(边界减一),

但此时这个最大堆又需要重新维护了,接着怎么做呢?

这时候就要立刻请出 HeapAdjust 大人,你要告诉它,你要调整的元素位置和它的边界。

让它把当前这个元素重新再调整为最大值,嗯,至此,感谢万能的 HeapAdjust 大人。

好吧,这就是堆排序的所有操作思想了.

至此,该篇已经结束,下面的是我个人的隐藏内容(/w \)(真有人会看到这里?没在开我玩笑吧~)

能看到这里的同学基本都是深爱着计算机的同学了,既然能看到这里,就让我来说点细节的问题吧~

其实,百度百科提供的代码很LOW的,下面就放出我的代码吧~

	void HeapAdjust(int arr[], int pos, int len)
{
int keypos = (pos << 1) + 1, KeyElement = arr[pos];//keypos 为其左子节点位置, KeyElement 为当前需要调整位置的元素.
while(keypos < len)//检查左子结点是否越界.
{
if (keypos + 1 < len && arr[keypos] < arr[keypos + 1]) keypos++;//若右结点存在且比左结点更大则替换
if (KeyElement > arr[keypos]) break;//若不存在比KeyElement更大的的子结点则中断调整位置
arr[pos] = arr[keypos], pos = keypos, keypos = (pos << 1) + 1;//将当前节点覆盖其父节点,同时更新当前结点为其子节点
}
arr[pos] = KeyElement;//最后确定位置后归位
}

到这里就是我的改进版,接着,还有牺牲空间 N 的进一步优化。

	while (true)
{
//因数组数据特殊性,当树结点单枝时一定存在左结点(pos + 1 >= max),则pos不变动
if(arr[pos] < arr[pos + 1] && pos + 1 < max) pos++;//若右结点存在且大于则替换为右结点
//将进一步移动到其子节点
arr[pos - 1 >> 1] = arr[pos], pos = (pos << 1) + 1;
//left and right node all is zero to break;
if (pos > max || (!arr[pos] && !arr[pos + 1]))
{
//assume zero is side element
arr[pos - 1 >> 1] = 0;
break;
}
}

目前还没发现有人和我做一样的改进诶嘿嘿,改进的原理为:

将堆顶元素抽走,接着下面的左右子结点较大的元素将被提上来,最后到了底(边界)时,将其赋值为0,在这之后,但凡两子节点为 0 (side element) 时将中断进一步对其子节点调整的操作。

理论上分析,这种牺牲空间N的做法比传统的(大量重复的 HeapAdjust )更为节省操作,代价就只需要浪费 N 的空间来存放数据.

这里小小提及一下现在C++的 STL sort()

它采用的是 QuickSort、 HeapSort 、 InsertionSort 的结合

总称 Introsort (内省排序)。

本文最后真挚地感谢 LYC 小队的 Y 同学日夜陪我研究和折腾,还有修改,表示没有他一同参与我感觉会很乏味的说,至于 L 同学则跑去打机了,先前 Astar 把他玩坏了,还是给他自由玩耍的空间好了= -=~

本想把改进的代码发布到 wiki 的,但现在想想还是留在这个没人关注的空间等到某天用上了再让它面世吧 =- = ~

两年后,嗯,当年的我学的都是些啥玩意!

说说 HeapSort 堆排序思想,以及个人优化方案。(老物)的更多相关文章

  1. Heapsort 堆排序算法详解(Java实现)

    Heapsort (堆排序)是最经典的排序算法之一,在google或者百度中搜一下可以搜到很多非常详细的解析.同样好的排序算法还有quicksort(快速排序)和merge sort(归并排序),选择 ...

  2. Adapter优化方案的探索

    概要:使用Adapter的注意事项与优化方案本文的例子都可以在结尾处的示例代码连接中看到并下载,如果喜欢请star,如果觉得有纰漏请提交issue,如果你有更好的点子可以提交pull request. ...

  3. c#并行任务多种优化方案分享(异步委托)

    遇到一个多线程任务优化的问题,现在解决了,分享如下. 假设有四个任务: 任务1:登陆验证(CheckUser) 任务2:验证成功后从Web服务获取数据(GetDataFromWeb) 任务3:验证成功 ...

  4. 五个Taurus垃圾回收compactor优化方案,减少系统资源占用

    简介 TaurusDB是一种基于MySQL的计算与存储分离架构的云原生数据库,一个集群中包含多个存储几点,每个存储节点包含多块磁盘,每块磁盘对应一个或者多个slicestore的内存逻辑结构来管理. ...

  5. C++高并发场景下读多写少的优化方案

    概述 一谈到高并发的优化方案,往往能想到模块水平拆分.数据库读写分离.分库分表,加缓存.加mq等,这些都是从系统架构上解决.单模块作为系统的组成单元,其性能好坏也能很大的影响整体性能,本文从单模块下读 ...

  6. Tomcat 配置详解/优化方案

     转自:http://blog.csdn.net/cicada688/article/details/14451541 Service.xml Server.xml配置文件用于对整个容器进行相关的配置 ...

  7. 一个网站完整详细的SEO优化方案

    根据自己的个人经验完成了这篇文章,希望对SEOer有点帮助,高手直接跳过,请勿喷水... 一个完整的SEO优化方案主要由四个小组组成: 一.前端/页编人员 二.内容编辑人员 三.推广人员 四.数据分析 ...

  8. mysql 性能优化方案

    网 上有不少MySQL 性能优化方案,不过,mysql的优化同sql server相比,更为麻烦与复杂,同样的设置,在不同的环境下 ,由于内存,访问量,读写频率,数据差异等等情况,可能会出现不同的结果 ...

  9. iOS界面跳转的一些优化方案

    原文地址: http://blog.startry.com/2016/02/14/Think-Of-UIViewController-Switch/ iOS界面跳转的一些优化方案 App应用程序开发, ...

随机推荐

  1. QtCreator常用之快捷键

    1. Ctrl(按住)+ Tab快速切换已打开的文件 2. 自动添加成员函数实体(.cpp)定义: 将光标移动到h文件中的方法声明. 按Alt(按住)+ Enter,再按回车键 将在cpp中添加该函数 ...

  2. [BZOJ2729]:[HNOI2012]排队(组合数学)

    题目传送门 题目描述 某中学有n名男同学,m名女同学和两名老师要排队参加体检.他们排成一条直线,并且任意两名女同学不能相邻,两名老师也不能相邻,那么一共有多少种排法呢?(注意:任意两个人都是不同的) ...

  3. java中的过滤器 --Filter

    package filter; import java.io.IOException; import javax.servlet.Filter; import javax.servlet.Filter ...

  4. nginx利用try_files实现多个源

    比如一个视频网站,视频资源分散在几台机器上,但是给用要提供统一的IP,路径,端口.就需要nginx,先检查本机是否有改文件,如果没有就代理其他地方 location / { root /data/ww ...

  5. NDK下编译JNI

    NDK环境下编译JNI 下载demo.tar.gz然后解压 弄个套路 1.编辑build.sh设置好NDK目录 2.把cpp文件放到code下面 运行sh build.sh即可

  6. webpack前置知识2(JavaScript项目初始化)

    所有的JavaScript项目都是在终端输入npm init -y进行项目初始化,如果要自定义项目规则,去掉 -y 参数. vscode终端快捷键ctrl+` 初始化 运行上述命令后,项目内会新建一个 ...

  7. hashMap与 hashTable , ArrayList与linkedList 的区别(详细)

    ArrayList 采用的是数组形式来保存对象的,这种方式将对象放在连续的位置中,所以最大的缺点就是插入删除时非常麻烦LinkedList 采用的将对象存放在独立的空间中,而且在每个空间中还保存下一个 ...

  8. 关于ajax请求controller返回中文乱码的解决方法!

    问题描述:前台ajax请求 Spring框架的 controller 返回的本应该是一段中文字符串 ,结果返回了一串 ?????? 解决办法:在RequestMapping的属性中添加返回数据类型属性 ...

  9. C++拷贝构造函数心得

    C++Primer作者提到拷贝构造函数调用的三种时机: 1. 当用一个类对象去初始化另外一个类对象(类似于 AClass aInstance = bInstance),这里不是调用赋值构造函数(也叫赋 ...

  10. 中国MOOC_零基础学Java语言_第6周 使用对象_2GPS数据处理

    2 GPS数据处理(5分) 题目内容: NMEA-0183协议是为了在不同的GPS(全球定位系统)导航设备中建立统一的BTCM(海事无线电技术委员会)标准,由美国国家海洋电子协会(NMEA-The N ...