基于PHP实现堆排序原理及实例详解


Posted in PHP onJune 19, 2020

堆(heap)是计算机科学中一类特殊的数据结构的统称,通常是一个可以被看做一棵树的数组对象。

堆{k1,k2,ki,…,kn} (ki <= k2i,ki <= k2i+1)|(ki >= k2i,ki >= k2i+1), (i = 1,2,3,4...n/2)

关于堆:

  • 堆中某个节点的值总是不大于或不小于其父节点的值;
  • 堆总是一棵完全二叉树(下面)。
  • 将根节点最大的堆叫做最大堆或大根堆,根节点最小的堆叫做最小堆或小根堆。

完全二叉树

说到堆排序,就不能不提完全二叉树,这些基本概念在网上到处都是,我摘了个最简单的。。

完全二叉树:除最后一层外,每一层上的节点数均达到最大值;在最后一层上只缺少右边的若干结点。

我自己总结认为,正是因为有下面两个特点,

  • 只允许最后一层有空缺结点且空缺在右边,即叶子结点只能在层次最大的两层上出现(存储方式的规则性);
  • 若i>1,tree的双亲为tree[i div 2](其父子结点值的规律性);

才使得其进行排序非常方便。

堆排序

堆排序求升序用大顶堆,求降序用小顶堆。

本例用求降序的小顶堆来解析。

堆排序步骤如下:

1、我们将数据(49、38、65、97、76、13、27、50)建立一个数组$arr;

2、用数组$arr建立一个小顶堆(主要步骤,会在代码注释里解释,下图是用一个数组建立小顶堆的过程);

3、将堆的根(最小的元素)与最后一个叶子交换,并将堆长度减一,跳到第二步;

4、重复2-3步,直到堆中只有一个结点,排序完成。

基于PHP实现堆排序原理及实例详解

堆排序的PHP实现

//因为是数组,下标从0开始,所以,下标为n根结点的左子结点为2n+1,右子结点为2n+2; 
//初始化值,建立初始堆
$arr=array(49,38,65,97,76,13,27,50);
$arrSize=count($arr);

//将第一次排序抽出来,因为最后一次排序不需要再交换值了。
buildHeap($arr,$arrSize);

for($i=$arrSize-1;$i>0;$i--){
  swap($arr,$i,0);
  $arrSize--;
  buildHeap($arr,$arrSize);  
}

//用数组建立最小堆
function buildHeap(&$arr,$arrSize){
  //计算出最开始的下标$index,如图,为数字"97"所在位置,比较每一个子树的父结点和子结点,将最小值存入父结点中
  //从$index处对一个树进行循环比较,形成最小堆
  for($index=intval($arrSize/2)-1; $index>=0; $index--){
    //如果有左节点,将其下标存进最小值$min
    if($index*2+1<$arrSize){
      $min=$index*2+1;
      //如果有右子结点,比较左右结点的大小,如果右子结点更小,将其结点的下标记录进最小值$min
      if($index*2+2<$arrSize){
        if($arr[$index*2+2]<$arr[$min]){
          $min=$index*2+2;
        }
      }
      //将子结点中较小的和父结点比较,若子结点较小,与父结点交换位置,同时更新较小
      if($arr[$min]<$arr[$index]){
        swap($arr,$min,$index);
      }  
    }
  }
}

//此函数用来交换下数组$arr中下标为$one和$another的数据
function swap(&$arr,$one,$another){
  $tmp=$arr[$one];
  $arr[$one]=$arr[$another];
  $arr[$another]=$tmp;
}

下面是排序的最终结果:

基于PHP实现堆排序原理及实例详解

堆用来进行全排序,时间复杂度是O(nlogn)

而快排用来全排序,平均时间复杂度也是O(nlogn)

但堆排序可以用来求 TopK 时,堆的时间复杂度为O(Klog2(n),因为它只需要进行 K 轮排序即可。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

PHP 相关文章推荐
PHP 提取图片img标记中的任意属性的简单实例
Dec 10 PHP
PHP中使用file_get_contents抓取网页中文乱码问题解决方法
Dec 17 PHP
Laravel中使用自己编写类库的3种方法
Feb 10 PHP
codeigniter显示所有脚本执行时间的方法
Mar 21 PHP
php示例详解Constructor Prototype Pattern 原型模式
Oct 15 PHP
mysql alter table命令修改表结构实例详解
Sep 24 PHP
Zend Framework动作控制器用法示例
Dec 09 PHP
PHP实现基于回溯法求解迷宫问题的方法详解
Aug 17 PHP
PHP简单实现二维数组赋值与遍历功能示例
Oct 19 PHP
PHP addslashes()函数讲解
Feb 03 PHP
PHP代码覆盖率统计详解
Jul 22 PHP
php字符串倒叙
Apr 01 PHP
深入分析PHP设计模式
Jun 15 #PHP
Laravel6.18.19如何优雅的切换发件账户
Jun 14 #PHP
Laravel服务容器绑定的几种方法总结
Jun 14 #PHP
Laravel如何实现适合Api的异常处理响应格式
Jun 14 #PHP
Yii redis集合的基本使用教程
Jun 14 #PHP
php获取小程序码的实现代码(B类接口)
Jun 13 #PHP
laravel admin实现分类树/模型树的示例代码
Jun 10 #PHP
You might like
模板引擎正则表达式调试小技巧
2011/07/20 PHP
8个PHP程序员常用的功能汇总
2014/12/18 PHP
laravel5使用freetds连接sql server的方法
2018/12/07 PHP
JS IE和FF兼容性问题汇总
2009/02/09 Javascript
jquery入门必备的基本认识及实例(整理)
2013/06/24 Javascript
jQuery学习笔记之总体架构
2014/06/03 Javascript
一个简单的jQuery插件ajaxfileupload.js实现ajax上传文件例子
2014/06/26 Javascript
Javascript动态创建div的方法
2015/02/09 Javascript
Nodejs基于LRU算法实现的缓存处理操作示例
2017/03/17 NodeJs
基于JavaScript实现的插入排序算法分析
2017/04/14 Javascript
Vue中img的src属性绑定与static文件夹实例
2017/05/18 Javascript
细说webpack源码之compile流程-rules参数处理技巧(1)
2017/12/26 Javascript
angularjs 获取默认选中的单选按钮的value方法
2018/02/28 Javascript
p5.js入门教程之图片加载
2018/03/20 Javascript
详解vue表单——小白速看
2018/04/08 Javascript
python实现模拟按键,自动翻页看u17漫画
2015/03/17 Python
Python字符串转换成浮点数函数分享
2015/07/24 Python
Python3中使用urllib的方法详解(header,代理,超时,认证,异常处理)
2016/09/21 Python
Python简单的制作图片验证码实例
2017/05/31 Python
Python基于列表模拟堆栈和队列功能示例
2018/01/05 Python
Python小进度条显示代码
2019/03/05 Python
Python 3.6打包成EXE可执行程序的实现
2019/10/18 Python
布隆过滤器的概述及Python实现方法
2019/12/08 Python
Python3爬虫中关于中文分词的详解
2020/07/29 Python
Dyson戴森波兰官网:Dyson.pl
2019/08/05 全球购物
Java里面有没有全局变量?为什么?
2015/02/06 面试题
九州传奇上机题
2014/07/10 面试题
国培计划培训感言
2014/03/11 职场文书
关于热爱祖国的演讲稿
2014/05/04 职场文书
教师党员批评与自我批评
2014/10/15 职场文书
小学中等生评语
2014/12/29 职场文书
严以律己专题学习研讨会发言材料
2015/11/09 职场文书
七年级作文之我的梦想
2019/10/16 职场文书
导游词之澳门妈祖庙
2019/12/19 职场文书
教你用python实现12306余票查询
2021/06/30 Python
十大公认最好看的动漫:《咒术回战》在榜,《钢之炼金术师》第一
2022/03/18 日漫