黑马程序员技术交流社区

标题: 【上海校区】堆排序(php实现) [打印本页]

作者: 不二晨    时间: 2018-11-16 09:40
标题: 【上海校区】堆排序(php实现)
简单的堆排序定义(wiki)
堆排序(Heapsort)是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。
简单思路有了前一篇博客实现的堆,自然而然的想到,将给出的数组先形成一个大根堆,然后逐一的取出,后逆序的返回即可。
代码实现<?php// require('../Library/SortTestHelper.php');require('../SortingAdvance/QuickSort.php');/** * 最大堆 */class MaxHeap{        private $data;        private $count;        public function __construct(){                $this->data = array();                $this->count = 0;        }        // public function __construct($arr){        // }        public function insert($item){                //从1开始        $this->data[$this->count + 1] = $item;        $this->_shiftUp($this->count+1);        $this->count++;    }    public function  extractMax(){        $ret = $this->data[1];        swap( $this->data, 1 , $this->count);        $this->count--;        $this->_shiftDown(1);        return $ret;    }    public function getMax(){        return $this->data[1];    }    public function isEmpty(){        return $this->count == 0;    }    public function getData(){            return $this->data;    }    /**     * [_shiftUp 新加入到堆中的元素直接放在数组后面,再与父元素比较后交换位置,直到根节点]     * @param  [type] $k [description]     * @return [type]    [description]     */        private function _shiftUp($k){                //如果叶子节点的值比父元素大交换位置,并更新k的值        while( $k > 1 && $this->data[(int)($k/2)] < $this->data[$k] ){            // swap( $this->data[(int)($k/2)], $this->data[$k] );            swap( $this->data, (int)($k/2) , $k);            $k = (int)($k/2);        }    }    /**     * [_shiftDown 元素出堆的时候,需要维护此时的堆依然是一个大根堆, 此时将数组元素的最后一个值与第一个值交换,后从上往下维护堆的性质]     * @param  [type] $k [description]     * @return [type]    [description]     */    private function _shiftDown($k){            //2k代表该节点的左子节点        while( 2*$k <= $this->count ){            $j = 2*$k;            //判断右节点是否存在,并且右节点大于左节点            if( $j+1 <= $this->count && $this->data[$j+1] > $this->data[$j] ) $j ++;            if( $this->data[$k] >= $this->data[$j] ) break;            // swap( $this->data[$k] , $this->data[$j] );            swap( $this->data, $k , $j );            $k = $j;        }    }}//简单的堆排序function headSort1(&$arr, $n){    $head_obj = new MaxHeap();    for ($i=0; $i < $n; $i++) {        $head_obj->insert($arr[$i]);    }    //逆序输出    for ($i=$n-1; $i >= 0; $i--) {        $arr[$i] = $head_obj -> extractMax();    }}$n = 10000;$arr = generateRandomArray($n, 0, $n);$copy_arr1 = $arr;$copy_arr2 = $arr;$copy_arr3 = $arr;$copy_arr4 = $arr;// $arr = generateNearlyOrderedArray($n, 100);testSort("headSort1", "headSort1", $arr, $n);testSort("mergeSort", "mergeSort", $copy_arr1, $n);testSort("quickSort", "quickSort", $copy_arr2, $n);testSort("quickSort2", "quickSort2", $copy_arr3, $n);testSort("quickSort3", "quickSort3", $copy_arr4, $n);?>复制代码结果
headSort1运行的时间为:0.70801091194153smergeSort运行的时间为:0.94017505645752squickSort运行的时间为:0.30204510688782squickSort2运行的时间为:0.19032001495361squickSort3运行的时间为:0.36022400856018s复制代码结果分析
for ($i=0; $i < $n; $i++) {        $head_obj->insert($arr[$i]);    }//逆序输出for ($i=$n-1; $i >= 0; $i--) {        $arr[$i] = $head_obj -> extractMax();}复制代码这段代码进行了两次O(NlogN)时间复杂度的运算,并且是将已有的数组去生成一个堆,然后借助这个堆重新赋值给数组,其实可以直接用数组去形成一个堆,然后我们就可以进行一次O(NlogN)复杂度的运算。我们只需要重写一个构造函数,通过一个数组去构造它!
堆排序的优化代码实现<?phprequire('../SortingAdvance/QuickSort.php');/** * 根据已知数组最大堆 */class HeapSort{        private $data;        private $count;        public function __construct($arr, $n){                $this->data = array();        $this->count = $n;        for ($i=0; $i < $n; $i++) {            //从1开始            $this->data[$i+1] = $arr[$i];        }        //叶子节点已经是一颗大根堆了,从最后一个非叶子节点进行_shiftDown,知道根节点        for ($i= (int)($n/2); $i >= 1 ; $i--) {            $this->_shiftDown($i);        }        }        // public function __construct($arr){        // }        public function insert($item){                //从1开始        $this->data[$this->count + 1] = $item;        $this->_shiftUp($this->count+1);        $this->count++;    }    public function  extractMax(){        $ret = $this->data[1];        swap( $this->data, 1 , $this->count);        $this->count--;        $this->_shiftDown(1);        return $ret;    }    public function getMax(){        return $this->data[1];    }    public function isEmpty(){        return $this->count == 0;    }    public function getData(){            return $this->data;    }    /**     * [_shiftUp 新加入到堆中的元素直接放在数组后面,再与父元素比较后交换位置,直到根节点]     * @param  [type] $k [description]     * @return [type]    [description]     */        private function _shiftUp($k){                //如果叶子节点的值比父元素大交换位置,并更新k的值        while( $k > 1 && $this->data[(int)($k/2)] < $this->data[$k] ){            // swap( $this->data[(int)($k/2)], $this->data[$k] );            swap( $this->data, (int)($k/2) , $k);            $k = (int)($k/2);        }    }    /**     * [_shiftDown 元素出堆的时候,需要维护此时的堆依然是一个大根堆, 此时将数组元素的最后一个值与第一个值交换,后从上往下维护堆的性质]     * @param  [type] $k [description]     * @return [type]    [description]     */    private function _shiftDown($k){            //2k代表该节点的左子节点        while( 2*$k <= $this->count ){            $j = 2*$k;            //判断右节点是否存在,并且右节点大于左节点            if( $j+1 <= $this->count && $this->data[$j+1] > $this->data[$j] ) $j ++;            if( $this->data[$k] >= $this->data[$j] ) break;            // swap( $this->data[$k] , $this->data[$j] );            swap( $this->data, $k , $j );            $k = $j;        }    }}function headSort2(&$arr, $n){    $head_obj = new HeapSort($arr, $n);    for ($i=$n-1; $i >= 0; $i--) {        $arr[$i] = $head_obj -> extractMax();    }}$n = 10000;$arr = generateRandomArray($n, 0, $n);$copy_arr1 = $arr;$copy_arr2 = $arr;$copy_arr3 = $arr;$copy_arr4 = $arr;testSort("headSort2", "headSort2", $arr, $n);testSort("mergeSort", "mergeSort", $copy_arr1, $n);testSort("quickSort", "quickSort", $copy_arr2, $n);testSort("quickSort2", "quickSort2", $copy_arr3, $n);testSort("quickSort3", "quickSort3", $copy_arr4, $n);?>复制代码结果
quickSort2运行的时间为:0.12423086166382squickSort3运行的时间为:0.051468849182129sheadSort2运行的时间为:0.033907890319824smergeSort运行的时间为:0.020761013031006squickSort运行的时间为:0.016165018081665squickSort2运行的时间为:0.01316499710083squickSort3运行的时间为:0.026669025421143s

【转载】
作者:吴军旗
链接:https://juejin.im/post/5a5eb2eff265da3e347b45d7




作者: 不二晨    时间: 2018-11-21 17:14

作者: 魔都黑马少年梦    时间: 2018-11-22 16:55

作者: 小影姐姐    时间: 2018-11-22 17:18





欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2