基于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下用rmdir实现删除目录的三种方法小结
Apr 20 PHP
PHP新手用的Insert和Update语句构造类
Mar 31 PHP
PHP number_format() 函数定义和用法
Jun 01 PHP
php使用递归函数实现数字累加的方法
Mar 16 PHP
PHP中使用curl入门教程
Jul 02 PHP
PHP实现抓取迅雷VIP账号的方法
Jul 30 PHP
四个常见html网页乱码问题及解决办法
Sep 08 PHP
php字符串的替换,分割和连接方法
May 23 PHP
php简单统计中文个数的方法
Sep 30 PHP
详解PHP使用Redis存储session时的一个Warning定位
Jul 05 PHP
php调用云片网接口发送短信的实现方法
Oct 25 PHP
php微信开发之音乐回复功能
Jun 14 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
php expects parameter 1 to be resource, array given 错误
2011/03/23 PHP
PHP中如何实现常用邮箱的基本判断
2014/01/07 PHP
php实现的css文件背景图片下载器代码
2014/11/11 PHP
Kindeditor编辑器添加图片上传水印功能(php代码)
2017/08/03 PHP
Javascript 遍历对象中的子对象
2009/07/03 Javascript
JS getAttribute和setAttribute(取得和设置属性)的使用介绍
2013/07/10 Javascript
原生JS可拖动弹窗效果实例代码
2013/11/09 Javascript
简单的JavaScript互斥锁分享
2014/02/02 Javascript
js中函数调用的两种常用方法使用介绍
2014/07/17 Javascript
Javascript 完美运动框架(逐行分析代码,让你轻松了运动的原理)
2015/01/23 Javascript
JavaScript中诡异的delete操作符
2015/03/12 Javascript
JavaScript中的原型prototype属性使用详解
2015/06/05 Javascript
全面解析多种Bootstrap图片轮播效果
2016/05/27 Javascript
Angular 应用技巧总结
2016/09/14 Javascript
hovertree插件实现二级树形菜单(简单实用)
2016/12/28 Javascript
从零学习node.js之详解异步控制工具async(八)
2017/02/27 Javascript
Angular.JS去掉访问路径URL中的#号详解
2017/03/30 Javascript
JavaScript实现区块链
2018/03/14 Javascript
JS简单实现查看文档创建日期、修改日期和文档大小的方法示例
2018/04/08 Javascript
使用淘宝镜像cnpm安装Vue.js的图文教程
2018/05/17 Javascript
微信小程序自定义组件components(代码详解)
2019/10/21 Javascript
js实现简单掷骰子小游戏
2019/10/24 Javascript
Python cookbook(数据结构与算法)通过公共键对字典列表排序算法示例
2018/03/15 Python
python3连接MySQL数据库实例详解
2018/05/24 Python
详解python之heapq模块及排序操作
2019/04/04 Python
Python队列、进程间通信、线程案例
2019/10/25 Python
python numpy数组复制使用实例解析
2020/01/10 Python
HTML5新增加的功能详解
2016/09/05 HTML / CSS
用canvas显示验证码的实现
2020/04/10 HTML / CSS
毕业生幼师求职自荐信
2013/10/01 职场文书
公司保密承诺书
2014/03/27 职场文书
冬季安全检查方案
2014/05/23 职场文书
抗洪救灾感谢信
2015/01/22 职场文书
仓库保管员岗位职责
2015/02/09 职场文书
2015年行政工作总结范文
2015/04/09 职场文书
创业计划书之寿司
2019/07/19 职场文书