八、堆

优先队列(Priority Queue): 特殊的“队列”,取出元素的顺序是依照元素的优先权(关键字)大小,而不是元素进入队列的先后顺序。

若采用数组或链表实现优先队列
数组:
插入一元素总是插入尾部 ~Θ(1)
删除一查找最大(或最小)关键字 ~Θ(n)
从数组中删去需要移动元素 ~O(n)
链表:
插入一元素总是插入链表的头部 ~Θ(1)
删除一查找最大(或最小)关键字 ~Θ(n)
删去结点 ~Θ(1)
有序数组:
插入 - 找到合适的位置 ~ O(n)或O(log2N)
移动元素并插入 ~O(n)
删除 - 删去最后一个元素 ~Θ(1)
有序链表:
插入- 找到合适的位置 ~O(n)
插入元素 ~Θ(1)
删除- 删除首元素或最后元素 ~Θ(1)

  • 我们用优先队列的完全二叉树表示

堆的两个特性:

  • 结构性:用数组表示的完全二叉树;
  • 有序性:任意结点的关键字是其子树所有节点的最大值(或最小值)
       “最大堆(MaxHeap)”,也称“大顶堆”:最大值
       “最小堆(MinHeap)”,也称“小顶堆”:最小值

最大堆:


最小堆:

typedef int ElementType;
typedef struct HeapStruct{
    ElementType *elements;
    int size; //堆的当前元素个数
    int capacity; //堆的最大容量
} *MaxHeap;
#define MaxData 10000
//创建最大堆
MaxHeap create(int MaxSize)
{  //创建容量为MaxSize的空间的最大堆
    MaxHeap h = malloc(sizeof(struct HeapStruct));
    h->elements = malloc((MaxSize + 1)*sizeof(ElementType));
    h->size = 0;
    h->capacity = MaxSize;
    //定义“哨兵”为大于堆中所有可能元素的值,以便于以后更快操作
    h->elements[0] = MaxData;
    return h;
}
//判断堆满
bool isFull(MaxHeap h)
{
    return h->size == h->capacity;
}
//判空
bool isEmpty(MaxHeap h)
{
    return h->size == 0;
}

最大堆插入

//往堆里插入元素 T=O(logN)
void insert(MaxHeap h, ElementType item)
{
    int i;
    if(isFull(h))
    {
        printf("最大堆已满");
        return;
    }
    
    i = ++h->size; //i指向插入后堆中最后一个元素的位置
    for (; h->elements[i/2] < item; i/=2) {
        h->elements[i] = h->elements[i/2];
    }
    
    h->elements[i] = item;
}

以上可见插入操作时间复杂度为T = O(logN)

最大堆删除

//删除最大值
ElementType deleteMax(MaxHeap h)
{
    if(isEmpty(h)){
        printf("最大堆已为空");
        return -1;
    }
    ElementType maxItem = h->elements[1];
    ElementType item = h->elements[h->size--];
    int parent = 1, child;
    for (; 2*parent <= h->size; parent = child) {
        child = 2*parent ;
        
        if(child != h->size && h->elements[child + 1] > h->elements[child])
            child++; //child指向左右子结点较大者
        if(h->elements[child] <= item)
            break;
        else{
            h->elements[parent] = h->elements[child];
        }
    }
    h->elements[parent] = item;
    return maxItem;
}

以上可见删除操作时间复杂度为T = O(logN)

最大堆的构建

如果我们用最大堆插入的方式构建最大堆,其时间复杂度为T = O(NlogN)。
我们可以先将数据输入到数组,然后再调整。其时间复杂度为T = O(N/2logN)。

void MaxHeapAdjust(MaxHeap h, int p);
//构建堆
MaxHeap buildMaxHeap()
{
    MaxHeap h = create(10);
    int arr[] = {49,38,65,97,76,13,34,27,49,11};
    for (int i = 1; i <= h->capacity; i++) {
        h->elements[i] = arr[i - 1];
        h->size++;
    }
    /**
     调整堆
     思路跟删除堆相同
     从 h->size/2 的位置往前依次调整堆
     */
    for(int i = h->size/2; i > 0; i--)
    {
        MaxHeapAdjust(h, i);
    }
    return h;
}
//调整堆
void MaxHeapAdjust(MaxHeap h, int p)
{
    int parent = p, child;
    int item = h->elements[p];
    for (; parent * 2 <= h->size; parent = child) {
        child = 2 * parent;
        if(child != h->size && h->elements[child + 1] > h->elements[child])
            child++;
        
        if(item >= h->elements[child])
            break;
        else
            h->elements[parent] = h->elements[child];
    }
    
    h->elements[parent] = item;
}

最大堆堆排序

我们直接调用删除堆,返回最大值的方式,就可以获取从大到小的排序。

//堆排序
void MaxHeapSort(MaxHeap h)
{
    printf("堆排序:");
    for (int i = 1; i <= h->capacity; i++) {
        ElementType item = deleteMax(h);
        printf("%d ",item);
    }
}

调用堆排序

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

推荐阅读更多精彩内容