小奥的学习笔记

  • Home
  • Learning & Working
    • Speech Enhancement Notes
    • Programming language
    • Computer & DL
    • MOOC
  • Life
    • Life Time
    • Thinking & Comprehension
    • Volunteer
    • Plan
    • Travel
  • Footprints
  • GuestBook
  • About
    • About Me
    • 个人履历
    • 隐私策略
  1. 首页
  2. Study-notes
  3. Programming language
  4. Algorithm
  5. 正文

最大堆和最小堆

2019年4月19日 783点热度 0人点赞 0条评论

最大堆和最小堆

一、堆树的定义堆树的定义如下:

  1. 堆树是一颗完全二叉树;
  2. 堆树中某个节点的值总是不大于或不小于其孩子节点的值;
  3. 堆树中每个节点的子树都是堆树。当父节点的键值总是大于或等于任何一个子节点的键值时为最大堆。 当父节点的键值总是小于或等于任何一个子节点的键值时为最小堆。如下图所示,第一个图为最大堆,第二个为最小堆。

二、堆树的操作以最大堆为例进行讲解,最小堆同理。

原始数据为a[] = {4, 1, 3, 2, 16, 9, 10, 14, 8, 7},采用顺序存储方式,对应的完全二叉树如下图所示:

  1. 构造最大堆在构造堆的基本思想就是:首先将每个叶子节点视为一个堆,再将每个叶子节点与其父节点一起构造成一个包含更多节点的对。所以,在构造堆的时候,首先需要找到最后一个节点的父节点,从这个节点开始构造最大堆;直到该节点前面所有分支节点都处理完毕,这样最大堆就构造完毕了。

假设树的节点个数为n,以1为下标开始编号,直到n结束。对于节点i,其父节点为i/2;左孩子节点为i2,右孩子节点为i2+1。最后一个节点的下标为n,其父节点的下标为n/2。 如下图所示,最后一个节点为7,其父节点为16,从16这个节点开始构造最大堆;构造完毕之后,转移到下一个父节点2,直到所有父节点都构造完毕。

注意:上面构造最大堆的时候,从底而上,一层一层地不断基于父节点构造最大堆,直至到达根节点完成整个最大堆的构造。

  1. 代码如下
//最大堆
//=============

#include <iostream>

using namespace std;

struct MaxHeap
{
    int *heap;  //数据元素的存放空间,下标从1开始存放元素,下标0存放临时数据
    int HeapSize;  //数据元素的个数
    int MaxSize;  //存放数据元素空间的大小
};

//初始化最大堆
void MaxHeapint(MaxHeap &H)  
{
    for (int i = H.HeapSize / 2; i > 0; i--)
    {
        H.heap[0] = H.heap[i];
        int son = i * 2;
        while (son <= H.HeapSize)
        {
            if (son < H.HeapSize && H.heap[son] < H.heap[son + 1])
                son++;
            if (H.heap[son] < H.heap[0])
                break;
            else
            {
                H.heap[son / 2] = H.heap[son];
                son *= 2;
            }
        }
        H.heap[son / 2] = H.heap[0];
    }
}

//最大堆中插入节点
//思想:先在堆的最后添加一个节点,然后沿着堆树上升。
void MaxHeapInsert(MaxHeap &H, int x)
{
    if (H.HeapSize == H.MaxSize)
        return;
    int i = ++H.HeapSize;
    while (i != 1 && x > H.heap[i / 2])
    {
        H.heap[i] = H.heap[i / 2];
        i = i / 2;
    }
    H.heap[i] = x;
}

//最大堆堆顶节点删除
//思想:将堆树的最后的节点提到根结点,然后删除最大值,然后再把新的根节点放到合适的位置。
void MaxHeapDelete(MaxHeap &H , int &top)
{
    int x;
    if (H.HeapSize == 0)
        return;
    top = H.heap[1];
    H.heap[0] = H.heap[H.HeapSize--];
    int i = 1, son = i * 2;
    while (son <= H.HeapSize)
    {
        if (son < H.HeapSize && H.heap[son] < H.heap[son + 1])
            son++;
        if (H.heap[0] >= H.heap[son])
            break;
        H.heap[i] = H.heap[son];
        i = son;
        son = son * 2;
    }
    H.heap[i] = H.heap[0];
    return;
}

int main(void)
{
    int top;
    int &top_1 = top;
    MaxHeap H;
    cout << "请输入数据的size: " << endl;
    cin >> H.HeapSize;
    H.heap = new int(H.HeapSize + 1);
    cout << "请按照完全二叉树输入数据:" << endl;
    for (int i = 0; i <= H.HeapSize; i++)
    {
        cin >> H.heap[i];  //我们这里默认H.heap[0]为临时数据,设为0
    }

    MaxHeapint(H);  //将二叉树转换为最大堆
    MaxHeapInsert(H, 3);
    MaxHeapDelete(H, top);

    for (int i = 1; i <= H.HeapSize; i++)
    {
        cout << H.heap[i] << "  ";
    }
    cout << endl << top;
    system("pause");
}

参考自:https://www.cnblogs.com/zf-blog/p/9010977.html

本作品采用 知识共享署名 4.0 国际许可协议 进行许可
标签: 最大堆 最小堆
最后更新:2019年4月19日

davidcheung

这个人很懒,什么都没留下

打赏 点赞
< 上一篇
下一篇 >

文章评论

razz evil exclaim smile redface biggrin eek confused idea lol mad twisted rolleyes wink cool arrow neutral cry mrgreen drooling persevering
取消回复

搜索
欢迎关注我的个人公众号
最新 热点 随机
最新 热点 随机
DEEPFILTERNET:一种基于深度滤波的全频带音频低复杂度语音增强框架 奥地利匈牙利九日游旅程 论文阅读之Study of the General Kalman Filter for Echo Cancellation 小奥看房之鸿荣源珈誉府 杭州往返旅途及西溪喜来登和万怡的体验报告 2022年的第一篇碎碎念
奥地利匈牙利九日游旅程论文阅读之Study of the General Kalman Filter for Echo CancellationDEEPFILTERNET:一种基于深度滤波的全频带音频低复杂度语音增强框架
世园会志愿者备战之知识储备(1) 换个风格,换个心情 《鸟哥的Linux私房菜》(基础篇)笔记整理(第17章)Part.1 每日一感0828:只有友谊才是最坚定的力量 山东2011年高一入学新生取消文理分科,2014高考改革 Python chapter 5 learning notes
标签聚合
python学习 Python 生活 鸟哥的linux私房菜 算法 学习 linux leetcode 高中 Java
最近评论
davidcheung 发布于 5 个月前(02月09日) The problem has been fixed. May I ask if you can s...
tk88 发布于 5 个月前(02月07日) Hmm is anyone else having problems with the pictur...
cuicui 发布于 9 个月前(10月20日) :wink:
niming 发布于 10 个月前(09月19日) 同级校友,能刷到太巧了
davidcheung 发布于 2 年前(08月16日) 我得找一下我之前整理的word文档看一下,如果找到了我就更新一下这篇文章。
Nolan 发布于 2 年前(07月25日) 您的笔记非常有帮助。贴图不显示了,可以更新一下吗?
davidcheung 发布于 3 年前(06月19日) 到没有看webrtc的代码。现在主要在看我们公司的代码了。。。只是偶尔看一看webrtc的东西。。。
aobai 发布于 3 年前(03月13日) gain_change_hangover_ 应该是每三个block 只能够调整一次,这样保证每帧...
匿名 发布于 5 年前(12月30日) 烫
小奥 发布于 5 年前(12月12日) webRTC里面的NS本身我记得就是在C++里面呀

COPYRIGHT © 2025 小奥的学习笔记. ALL RIGHTS RESERVED.

Theme Kratos Made By Seaton Jiang

陕ICP备19003234号-1

鲁公网安备37120202000100号