基于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 相关文章推荐
ThinkPHP CURD方法之limit方法详解
Jun 18 PHP
Laravel框架数据库CURD操作、连贯操作总结
Sep 03 PHP
PHP图片自动裁切应付不同尺寸的显示
Oct 16 PHP
php实现MySQL数据库备份与还原类实例
Dec 09 PHP
ThinkPHP中数据操作案例分析
Sep 27 PHP
Yii基于数组和对象的Model查询技巧实例详解
Dec 28 PHP
php微信开发自定义菜单
Aug 27 PHP
PHP框架自动加载类文件原理详解
Jun 06 PHP
PHP判断json格式是否正确的实现代码
Sep 20 PHP
Laravel程序架构设计思路之使用动作类
Jun 07 PHP
php unlink()函数使用教程
Jul 12 PHP
imagettftext() 失效,不起作用
Mar 09 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
Terran兵种介绍
2020/03/14 星际争霸
十天学会php之第四天
2006/10/09 PHP
php学习之数据类型之间的转换代码
2011/05/29 PHP
PHP删除目录及目录下所有文件的方法详解
2013/06/06 PHP
通过curl模拟post和get方式提交的表单类
2014/04/23 PHP
PHP从尾到头打印链表实例讲解
2018/09/27 PHP
PHP生成随机密码4种方法及性能对比
2020/12/11 PHP
jQuery ajax dataType值为text json探索分享
2013/09/23 Javascript
用javascript对一个json数组深度赋值示例
2014/07/27 Javascript
深入探讨javascript函数式编程
2015/10/11 Javascript
jquery单击事件和双击事件冲突解决方案
2016/03/02 Javascript
js实现密码强度检测【附示例】
2016/03/30 Javascript
浅谈AngularJs指令之scope属性详解
2016/10/24 Javascript
详解vue slot插槽的使用方法
2017/06/13 Javascript
基于node.js制作简单爬虫教程
2017/06/29 Javascript
React Native 集成jpush-react-native的示例代码
2017/08/16 Javascript
在原生不支持的旧环境中添加兼容的Object.keys实现方法
2017/09/11 Javascript
create-react-app修改为多页面支持的方法
2018/05/17 Javascript
[02:28]DOTA2英雄基础教程 狼人
2013/12/23 DOTA
[01:03:36]Ti4 循环赛第三日DK vs Titan
2014/07/12 DOTA
[00:52]DOTA2第二届亚洲邀请赛预选赛宣传片
2017/01/13 DOTA
更改Python命令行交互提示符的方法
2015/01/14 Python
Python制作简易注册登录系统
2016/12/15 Python
Python实现的爬虫功能代码
2017/06/24 Python
python logging重复记录日志问题的解决方法
2018/07/12 Python
python基础知识(一)变量与简单数据类型详解
2019/04/17 Python
Python列表(list)所有元素的同一操作解析
2019/08/01 Python
python虚拟环境的安装和配置(virtualenv,virtualenvwrapper)
2019/08/09 Python
用python对excel进行操作(读,写,修改)
2020/12/25 Python
StubHub新西兰:购买和出售你的门票
2019/04/22 全球购物
颇特女士:NET-A-PORTER(直邮中国)
2020/07/11 全球购物
不错的求职信范文
2014/07/20 职场文书
党的群众路线教育实践活动个人对照检查材料(企业)
2014/11/05 职场文书
2014年政工师工作总结
2014/12/18 职场文书
MySQL索引失效的典型案例
2021/06/05 MySQL
手把手教你导入Go语言第三方库
2021/08/04 Golang