堆排序

参考文章:图解排序算法(三)之堆排序

说在前面:本来对堆这种数据结构不了解,然后直接看的堆排序的介绍,看完之后一脸懵逼。。。这个堆是咋构建的?这个 “i*2+1” 是什么玩意???于是重新查找,终于看到了参考文章,里面详细又清楚的介绍了堆,以及堆排序,非常容易理解,也非常感谢这哥们儿,所以我一定要贴出他的这篇文章。如果看我写的还是不能理解,相信我,去看看他的那篇文章,你一定会有收获。

一、几个重要概念

  1. 二叉树: 是每个结点最多有两个子树的树结构。
  2. 完全二叉树:是除最后一层外,其余层都是满的,或最后一层或者是满的,或者是在右边缺少连续若干节点的二叉树。
  3. 堆:是具有以下性质的完全二叉树,
    ① 每个结点的值都大于或等于其左右孩子结点的值,称为大顶堆;
    ② 或者每个结点的值都小于或等于其左右孩子结点的值,称为小顶堆。
    如图所示:

如果,我们对堆中的结点按层进行编号,将这种逻辑结构映射到数组中就是下面这个样子:

该数组从逻辑上讲就是一个堆结构,我们用简单的公式来描述一下堆的定义就是:
假设某个节点下标为i,节点值为arr[i],则这个节点的左子节点为arr[2i+1],又子节点为arr[2i+2],那么
大顶堆中满足条件:arr[i] >= arr[2i+1] && arr[i] >= arr[2i+2]
小顶堆中满足条件:arr[i] <= arr[2i+1] && arr[i] <= arr[2i+2]

二、堆排序思想

将待排序序列构造成一个大顶堆,此时,整个序列的最大值就是堆顶的根节点。将其与末尾元素进行交换,此时末尾就为最大值。然后将剩余n-1个元素重新构造成一个堆,这样会得到n个元素的次小值。如此反复执行,便能得到一个有序序列了。

三、算法描述

  • 1.由输入的无序数组构造一个最大堆,作为初始的无序区
  • 2.把堆顶元素(最大值)和堆尾元素互换
  • 3.把堆(无序区)的尺寸缩小1,并调用heapify(A, 0)从新的堆顶元素开始进行堆调整
  • 4.重复步骤2,直到堆的尺寸为1

四、代码实现

<?php
//输出数组
function echo_arr($arr, $sym=' '){
    echo implode($sym, $arr).'<br>';
}

//交换数字
function swap_item($arr, $i, $j){
    $tmp = $arr[$i];
    $arr[$i] = $arr[$j];
    $arr[$j] = $tmp;
    return $arr;
}

//检测数组
function check_arr($arr){
    if(!is_array($arr)){
        return 'arr error';
    }

    $len = count($arr);
    if($len <= 1){
        return $arr;
    }

    return true;
}

//计算耗时
function time_consuming($start, $end, $name=''){
    $t = $end - $start;
    var_dump($name.'耗时:'.$t);
}

//显示起始时间
function show_time($start, $end, $name=''){
    var_dump($name.' start:'.$start);
    var_dump($name.' end  :'.$end);
}

//获取乱序的数组
function get_test_arr($len = 200){
    for( $i = 0; $i < $len; $i++ ){
      $arr[] = mt_rand( 1,1000);
    }

    return $arr;
}

//获取顺序的数组
function get_sort_arr($len = 200){
    for( $i = 0; $i < $len; $i++ ){
      $arr[] = $i;
    }

    return $arr;
}

//排序主程序
function sort_main($arr){
    $check_re = check_arr($arr);

    if($check_re === true){
        $start = microtime(true);
        heap_sort($arr);
        $end = microtime(true);
        time_consuming($start, $end, 'quick_sort');
    }else{
        echo 'error';
        var_dump($arr);
    }
}


// $arr = array( 6, 4, 7, 2, 9, 8, 1 );
// $arr = [ 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 4, 4, 4, 4, 4, 4, 4, 4, 5, 6, 6, 6, 6, 6, 7 ];
// $arr = [];
// $arr = '123';

$arr = get_test_arr();
// $arr = get_sort_arr();
echo_arr($arr);
// var_dump($arr);
sort_main($arr);


//堆排序
function heap_sort($arr){
    $len = count($arr);
    //1.构建大顶堆
    for ($i= $len>>2-1; $i >= 0; $i--) { 
        //从第一个非叶子结点从下至上,从右至左调整结构
        $arr = adjust_heap($arr, $i, $len);
    }
    //2.调整堆结构+交换堆顶元素与末尾元素
    for ($j=$len - 1; $j > 0; $j--) {
        //将堆顶元素与末尾元素进行交换 
        $arr = swap_item($arr, 0, $j);
        //重新对堆进行调整
        $arr = adjust_heap($arr, 0, $j);
    }
    echo_arr($arr);
}

//调整堆
function adjust_heap($arr, $cur, $len){
    //获取当前节点
    $tmp = $arr[$cur];
    //从当前节点的左子节点开始
    for ($i = $cur*2+1; $i < $len; $i = $i*2+1) { 
        //如果左子结点小于右子结点,i指向右子结点
        if($i+1 < $len && $arr[$i] < $arr[$i+1]){
            $i++;
        }
        //如果子节点大于当前节点,将子节点值赋给当前节点(不用进行交换!!!)
        if($arr[$i] > $tmp){
            $arr[$cur] = $arr[$i];
            $cur = $i;
        }else{
            break;
        }
    }
    $arr[$cur] = $tmp;

    return $arr;
}

//错误的堆调整方法
function error_adjust_heap($arr, $cur, $len){
    $tmp = $arr[$cur];
    for ($i = $cur*2+1; $i < $len; $i = $i*2+1) { 
        if($i+1 < $len && $arr[$i] < $arr[$i+1]){
            $i++;
        }
        if($arr[$i] > $tmp){
            $arr = swap_item($arr, $cur, $i);
        }else{
            break;
        }
    }

    return $arr;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容