(一)堆排序法

一、简介

二叉堆是完全二叉树或者是近似完全二叉树。

二叉堆满足二个特性:
1.父结点的键值总是大于或等于(小于或等于)任何一个子节点的键值
2.每个结点的左子树和右子树都是一个二叉堆(都是最大堆或最小堆)
当父结点的键值总是大于或等于任何一个子节点的键值时为最大堆。当父结点的键值总是小于或等于任何一个子节点的键值时为最小堆

堆排序是一种树形选择排序,是对直接选择排序的有效改进。

堆的定义下:具有n个元素的序列 (h1,h2,...,hn),当且仅当满足(hi>=h2i,hi>=2i+1)或(hi<=h2i,hi<=2i+1) (i=1,2,...,n/2)时称之为堆。在这里只讨论满足前者条件的堆。由堆的定义可以看出,堆顶元素(即第一个元素)必为最大项(大顶堆)。完全二叉树可以很直观地表示堆的结构。堆顶为根,其它为左子树、右子树。

思想:初始时把要排序的数的序列看作是一棵顺序存储的二叉树,调整它们的存储序,使之成为一个堆,这时堆的根节点的数最大。然后将根节点与堆的最后一个节点交换。然后对前面(n-1)个数重新调整使之成为堆。依此类推,直到只有两个节点的堆,并对它们作交换,最后得到有n个节点的有序序列。从算法描述来看,堆排序需要两个过程,一是建立堆,二是堆顶与堆的最后一个元素交换位置。所以堆排序有两个函数组成。一是建堆的渗透函数,二是反复调用渗透函数实现排序的函数。

二、步骤

  1. 产生初始堆
  2. 把堆顶的最大数取出,将剩余的堆继续调整为最大堆,以递归实现
  3. 剩余部分调整为最大堆后,再次将堆顶的最大数取出,再将剩余部分调整为最大堆,这个过程持续到剩余数只有一个时结束

三、示例


初始堆的构建

堆排序


四、代码实现

1. ShiftUp

// 将元素向上移动(将k位置的元素与父节点交换位置)
void shiftUp(int k) {
    while (k > 1 && data[k / 2] < data[k]) {    // data[k/2]是父节点,data[k]是该元素
        swap(data[k / 2], data[k]);
        k /= 2;     //更新一下k,以看与新的父节点之间是否违背了最大堆的定义。保证k最多到2,父节点k/2就是1
    }
}

2. ShiftDown

// 左(j)右(j+1)两个孩子比较,谁大就和父节点k交换,向下进行转换
void shiftDown(int k) {
    while (2 * k <= count) {    // 怎么表示k位置元素有孩子?在完全二叉树中只要有左孩子就表示有孩子了
        int j = 2 * k;  // 如果没有右孩子.在此轮循环中,data[k]和data[j]交换位置
        if (j + 1 <= count && data[j + 1] > data[j]) j++; // j + 1 <= count说明有右孩子   并且   右孩子比左孩子大   则j+=1
        if (data[k] >= data[j]) break;  // 如果父节点>=两个孩子就跳出循环,否则data[k]和data[j]进行交换
        swap(data[k], data[j]);
        k = j;  // 以后继续移动k位置的元素
    }
}

3. insert

// 向堆中添加一个元素。将新加入的元素放置到合适的位置(与它的父节点比大小,看是否违背了最大堆的定义)
void insert(Item item) {
    assert(count + 1 <= capacity);
    data[count + 1] = item;
    shiftUp(count + 1);
    count++;
}

4. extractMax

// 从堆中取出最大值,然后count-=1。取出后要把最后一个元素填补到该位置。最后调整位置保证最大堆定义
Item extractMax() {
    assert(count > 0);
    Item ret = data[1]; // 取出最大值
    swap(data[1], data[count]); // 将最大值元素与最后一个元素交换
    count--;    // 不考虑count位置的元素了
    shiftDown(1);   // 将第一个位置的元素向下移到合适的位置以满足最大堆的定义
    return ret;
}

5. 排序函数1

template<typename T>
void heapSort1(T arr[], int n) {
    MaxHeap<T> maxheap = MaxHeap<T>(n);
    for (int i = 0; i < n; i++)
        maxheap.insert(arr[i]);

    for (int i = n - 1; i >= 0; i--)    // 反向遍历,这样排序出来是递增顺序
        arr[i] = maxheap.extractMax();

}

6. 排序函数2

template<typename T>
void heapSort2(T arr[], int n){

    MaxHeap<T> maxheap = MaxHeap<T>(arr,n); // 已经构造成了一个堆,更快
    for( int i = n-1 ; i >= 0 ; i-- )
        arr[i] = maxheap.extractMax();

}

完整代码

#include <iostream>
#include <algorithm>
#include <cassert>
using namespace std;

template<typename Item>
class MaxHeap {
private:
    Item *data;
    int count;
    int capacity;

    // 将元素向上移动(将k位置的元素与父节点交换位置)
    void shiftUp(int k) {
        while (k > 1 && data[k / 2] < data[k]) {    // data[k/2]是父节点,data[k]是该元素
            swap(data[k / 2], data[k]);
            k /= 2;     //更新一下k,以看与新的父节点之间是否违背了最大堆的定义。保证k最多到2,父节点k/2就是1
        }
    }

    // 左(j)右(j+1)两个孩子比较,谁大就和父节点k交换,向下进行转换
    void shiftDown(int k) {
        while (2 * k <= count) {    // 怎么表示k位置元素有孩子?在完全二叉树中只要有左孩子就表示有孩子了
            int j = 2 * k;  // 如果没有右孩子.在此轮循环中,data[k]和data[j]交换位置
            if (j + 1 <= count && data[j + 1] > data[j]) j++; // j + 1 <= count说明有右孩子   并且   右孩子比左孩子大   则j+=1
            if (data[k] >= data[j]) break;  // 如果父节点>=两个孩子就跳出循环,否则data[k]和data[j]进行交换
            swap(data[k], data[j]);
            k = j;  // 以后继续移动k位置的元素
        }
    }

public:
    // 构造函数
    MaxHeap(int capacity) {
        data = new Item[capacity + 1];
        count = 0;
        this->capacity = capacity;
    }
    MaxHeap(Item arr[], int n) {
        data = new Item[n + 1];
        capacity = n;

        for (int i = 0; i < n; i++)
            data[i + 1] = arr[i];
        count = n;

        for (int i = count / 2; i >= 1; i--)
            shiftDown(i);
    }
    ~MaxHeap() {
        delete[] data;
    }
    int size() {
        return count;
    }
    bool isEmpty() {
        return count == 0;
    }

    // 向堆中添加一个元素。将新加入的元素放置到合适的位置(与它的父节点比大小,看是否违背了最大堆的定义)
    void insert(Item item) {
        assert(count + 1 <= capacity);
        data[count + 1] = item;
        shiftUp(count + 1);
        count++;
    }

    // 从堆中取出最大值,然后count-=1。取出后要把最后一个元素填补到该位置。最后调整位置保证最大堆定义
    Item extractMax() {
        assert(count > 0);
        Item ret = data[1]; // 取出最大值
        swap(data[1], data[count]); // 将最大值元素与最后一个元素交换
        count--;    // 不考虑count位置的元素了
        shiftDown(1);   // 将第一个位置的元素向下移到合适的位置以满足最大堆的定义
        return ret;
    }

    Item getMax() {
        assert(count > 0);
        return data[1];
    }
};

template<typename T>
void heapSort2(T arr[], int n) {

    MaxHeap<T> maxheap = MaxHeap<T>(arr, n);
    for (int i = n - 1; i >= 0; i--)
        arr[i] = maxheap.extractMax();

}

template<typename T>
void heapSort1(T arr[], int n) {

    MaxHeap<T> maxheap = MaxHeap<T>(n);
    for (int i = 0; i < n; i++)
        maxheap.insert(arr[i]);

    for (int i = n - 1; i >= 0; i--)
        arr[i] = maxheap.extractMax();

}

template<typename T>
void heapSort(T arr[], int n) {

    // 从(最后一个元素的索引-1)/2开始
    // 最后一个元素的索引 = n-1
    for (int i = (n - 1 - 1) / 2; i >= 0; i--)
        __shiftDown2(arr, n, i);

    for (int i = n - 1; i > 0; i--) {
        swap(arr[0], arr[i]);
        __shiftDown2(arr, i, 0);
    }
}

int main()
{
    int a[10] = { 3,2,5,21,9,10,7,16,8,20 };
    heapSort(a,10);
    for (int i = 0; i < 10; i++) {
        cout << a[i] << " ";
    }
}

五、评价

由于每次重新恢复堆的时间复杂度为O(logN),共N 1次重新恢复堆操作,再加上前面建立堆时N/2次向下调整,每次调整时间复杂度也为O(logN)。二次操作时间相加还是O(N logN)。故堆排序的时间复杂度为O(N logN)。

经典排序算法 - 堆排序Heap sort
经典排序算法系列7----堆与堆排序
选择排序:堆排序(Heap Sort)
白话经典算法系列之七 堆与堆排序

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

推荐阅读更多精彩内容

  • 第一章 绪论 什么是数据结构? 数据结构的定义:数据结构是相互之间存在一种或多种特定关系的数据元素的集合。 第二章...
    SeanCheney阅读 5,771评论 0 19
  • B树的定义 一棵m阶的B树满足下列条件: 树中每个结点至多有m个孩子。 除根结点和叶子结点外,其它每个结点至少有m...
    文档随手记阅读 13,218评论 0 25
  • 关于最大堆 什么是最大堆和最小堆?最大(小)堆是指在树中,存在一个结点而且该结点有儿子结点,该结点的data域值都...
    凌云壮志几多愁阅读 88,050评论 33 71
  • 生产者生产和消费者消费不一定会一一对应,为了解决这个问题引入缓存概念(容器)。生产者生产的产品,放在容器中,消费者...
    菩提大师阅读 432评论 1 1
  • 小西妈百人工程201705期206号Mason打卡第2天 1.听:0a unit 1-unit 12泛听 2.阅读...
    LELE_2855阅读 145评论 0 0