基于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 相关文章推荐
解决MySQL中文输出变成问号的问题
Jun 05 PHP
php中通过curl模拟登陆discuz论坛的实现代码
Feb 16 PHP
将博客园(cnblogs.com)数据导入到wordpress的代码
Jan 06 PHP
Zend Studio 实用快捷键一览表(精心整理)
Aug 10 PHP
php jquery 多文件上传简单实例
Dec 23 PHP
PHPMailer的主要功能特点和简单使用说明
Feb 17 PHP
ThinkPHP CURD方法之order方法详解
Jun 18 PHP
php表单提交与$_POST实例分析
Jan 26 PHP
PHP页面跳转实现延时跳转的方法
Dec 10 PHP
PHP中函数gzuncompress无法使用的解决方法
Mar 02 PHP
php实现基于PDO的预处理示例
Mar 28 PHP
PHP面向对象程序设计之多态性的应用示例
Dec 19 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 header示例代码(推荐)
2010/09/08 PHP
gd库图片下载类实现下载网页所有图片的php代码
2012/08/20 PHP
关于更改Zend Studio/Eclipse代码风格主题的介绍
2013/06/23 PHP
10个超级有用的PHP代码片段果断收藏
2015/09/23 PHP
php中的抽象方法和抽象类
2017/02/14 PHP
js 调整select 位置的函数
2008/02/21 Javascript
Mootools 1.2教程 输入过滤第一部分(数字)
2009/09/15 Javascript
解析John Resig Simple JavaScript Inheritance代码
2012/12/03 Javascript
使用AOP改善javascript代码
2015/05/01 Javascript
JavaScript严格模式详解
2015/11/18 Javascript
Jquery AJAX POST与GET之间的区别详细介绍
2016/10/17 Javascript
Javascript设计模式之装饰者模式详解篇
2017/01/17 Javascript
正则表达式基本语法及表单验证操作详解【基于JS】
2017/04/07 Javascript
Angularjs中使用轮播图指令swiper
2017/05/30 Javascript
简单实现js上传文件功能
2017/08/21 Javascript
vue.js模仿京东省市区三级联动的选择组件实例代码
2017/11/22 Javascript
redux中间件之redux-thunk的具体使用
2018/04/17 Javascript
使用vue-router beforEach实现判断用户登录跳转路由筛选功能
2018/06/25 Javascript
Angular6 写一个简单的Select组件示例
2018/08/20 Javascript
微信小程序与公众号实现数据互通的方法
2019/07/25 Javascript
Vue中keep-alive的两种应用方式
2020/07/15 Javascript
python支持断点续传的多线程下载示例
2014/01/16 Python
Python基于动态规划算法计算单词距离
2015/07/25 Python
python中执行shell的两种方法总结
2017/01/10 Python
Python中最大最小赋值小技巧(分享)
2017/12/23 Python
Python设计模式之MVC模式简单示例
2018/01/10 Python
使用Python paramiko模块利用多线程实现ssh并发执行操作
2019/12/05 Python
jupyter notebook远程访问不了的问题解决方法
2021/01/11 Python
CSS3的常见transformation图形变化用法小结
2016/05/13 HTML / CSS
旧时光糖果:Old Time Candy
2018/02/05 全球购物
Ruby如何创建一个线程
2013/03/10 面试题
房地产出纳岗位职责
2013/12/01 职场文书
党员示范岗材料
2014/12/19 职场文书
2016中秋节问候语
2015/11/11 职场文书
MySQL Shell import_table数据导入的实现
2021/08/07 MySQL
SpringBoot集成Redis的思路详解
2021/10/16 Redis