上次用Java实现了最大堆的封装,这次就来写一下最小堆的实现吧


插入函数的思路:

向堆中插入元素有两种情况,一种是堆为空,那么就让插入值作为根节点即可;另一种是堆不为空,那么此时就要进行判断当前节点与其父节点的大小关系比较。此时仍有两种情况,一种是当前节点大于父节点,这样正是我们所希望的;另一种是当前节点的值小于父节点,那么就要将二者的值进行调换,然后记得更新当前节点为原来父节点的位置,而父节点的位置同样需要更新(循环正常终止的时候说明已经到了根节点,此时最小值必定为根节点)

 bool Insert(T data){
        if(currentPos==MaxSize){
            cout<<"Sorry , this heap is full!"<<endl;
            return false;
        }
        currentPos++;
        int targetPos=currentPos-1;
        heap[targetPos]=data;
        while(targetPos>0){
            int parentPos=(targetPos-1)/2;
            if(heap[parentPos]<heap[targetPos]){
                break;
            }else{
                heap[targetPos]=heap[parentPos];
                targetPos=parentPos;
            }
        }
        return true;
    }
    //存在的bug是对根节点的大小比较,因为有可能targetPos=0而退出,此时就缺少了一次比较

siftDown调整过程思路:

给定要进行调整的节点的下标,我们只需要让它和它的两个子节点中最小的那个比较即可(前提是当前节点不是叶子节点),需要注意的是要先保存当前节点的值,比较之后按大小调整顺序即可。

 void siftDown(int siftPos){

        int siftPosition=siftPos;
        T temp=heap[siftPosition];

        int minChildPos=2*siftPosition+1;
        while(minChildPos<currentPos){          //保证比较的条件成立
            if((minChildPos<currentPos-1)&&(heap[minChildPos]>heap[minChildPos+1])){
                minChildPos++;
            }
            if(temp<heap[minChildPos]){
                break;
            }else{
                heap[siftPosition]=heap[minChildPos];
                siftPosition=minChildPos;
                minChildPos=2*siftPosition+1;
            }
        }
        //作用:当要进行调换的位置不满足循环要求时,说明要进行调换的位置是叶子节点,那就不需要变换咯(这里也包括正常比较情况,可正常使用)
        heap[siftPosition]=temp;
 }

删除对顶元素

需要注意的是currentPos的大小要实时的进行更新,然后返回所删除的堆顶元素即可

 T& deleteTop(){
        if(currentPos<0){
            cout<<"Sorry ,this heap is empty!"<<endl;
        }
        T target=heap[0];
        heap[0]=heap[currentPos-1];
        currentPos--;
        siftDown(0);
        return target;
    }

下面是完整的C++关于最小堆的实现的代码

#include <iostream>

using namespace std;
template<class T>
class MinHeap{
    T *heap;
    int MaxSize;
    int currentPos;

public:
    MinHeap(int MS){
        heap=new T[MS];
        currentPos=0;
        MaxSize=MS;
    }

    bool Insert(T data){
        if(currentPos==MaxSize){
            cout<<"Sorry , this heap is full!"<<endl;
            return false;
        }
        currentPos++;
        int targetPos=currentPos-1;
        heap[targetPos]=data;
        while(targetPos>0){
            int parentPos=(targetPos-1)/2;
            if(heap[parentPos]<heap[targetPos]){
                break;
            }else{
                heap[targetPos]=heap[parentPos];
                targetPos=parentPos;
            }
        }
        return true;
    }

    void siftDown(int siftPos){

        int siftPosition=siftPos;
        T temp=heap[siftPosition];

        int minChildPos=2*siftPosition+1;
        while(minChildPos<currentPos){          //保证比较的条件成立
            if((minChildPos<currentPos-1)&&(heap[minChildPos]>heap[minChildPos+1])){
                minChildPos++;
            }
            if(temp<heap[minChildPos]){
                break;
            }else{
                heap[siftPosition]=heap[minChildPos];
                siftPosition=minChildPos;
                minChildPos=2*siftPosition+1;
            }
        }
        //作用:当要进行调换的位置不满足循环要求时,说明要进行调换的位置是叶子节点,那就不需要变换咯
        heap[siftPosition]=temp;

        ////////////////////////////////////////////

    }

    T& deleteTop(){
        if(currentPos<0){
            cout<<"Sorry ,this heap is empty!"<<endl;
        }
        T target=heap[0];
        heap[0]=heap[currentPos-1];
        currentPos--;
        siftDown(0);
        return target;
    }
};

int main()
{
    cout << "Hello world!" << endl;
    MinHeap<int> minHeap(7);
    minHeap.Insert(1);
    minHeap.Insert(2);
    minHeap.Insert(4);
    minHeap.Insert(3);
    minHeap.Insert(6);
    minHeap.Insert(7);
    minHeap.Insert(5);
    for(int i=1;i<=7;i++){
        cout<<minHeap.deleteTop()<<endl;
    }
    return 0;
}

程序运行结果如下


总结:

代码中存在一定得错误,出在 Insert函数中。个人认为需要对targetPos为0的特殊情况再加一层判断,估计就能解决。但是对正常添加元素还是能得到比较正常的结果的。

C++实现最小堆及插入,调整顺序,删除堆顶元素的操作的更多相关文章

  1. Java实现堆的封装,进行插入,调整,删除堆顶以完成堆排序实例

    简介 堆对于排序算法是一个比较常用的数据结构,下面我就使用Java语言来实现这一算法 首先,我们需要知道堆的数据结构的形式,其实就是一个特殊的二叉树.但是这个二叉树有一定的特点,除了是完全二叉树以外, ...

  2. 笔试算法题(46):简介 - 二叉堆 & 二项树 & 二项堆 & 斐波那契堆

    二叉堆(Binary Heap) 二叉堆是完全二叉树(或者近似完全二叉树):其满足堆的特性:父节点的值>=(<=)任何一个子节点的键值,并且每个左子树或者右子树都是一 个二叉堆(最小堆或者 ...

  3. PTA 最小堆插入元素和删除堆顶(无哨兵元素) (20分)

    PTA 最小堆插入元素和删除堆顶(无哨兵元素) (20分) 对于给定的最小堆(优先队列),分别实现插入元素和删除堆顶的函数. 函数接口定义: int insertIntoHeap(struct Hea ...

  4. 堆+建堆、插入、删除、排序+java实现

    package testpackage; import java.util.Arrays; public class Heap { //建立大顶堆 public static void buildMa ...

  5. Kafka集群优化篇-调整broker的堆内存(heap)案例实操

    Kafka集群优化篇-调整broker的堆内存(heap)案例实操 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.查看kafka集群的broker的堆内存使用情况 1>. ...

  6. 设顺序表中的数据元素递增有序,试着写一算法,将x插入到顺序表上的适当位置上,以保持该表的有序性。

    原创,转载请注明出处.https://www.cnblogs.com/yangf428/p/11254370.html 天勤例题[2-1]: 设顺序表va中的数据元素递增有序.试写一算法,将x插入到顺 ...

  7. 堆实战(动态数据流求top k大元素,动态数据流求中位数)

    动态数据集合中求top k大元素 第1大,第2大 ...第k大 k是这群体里最小的 所以要建立个小顶堆 只需要维护一个大小为k的小顶堆 即可 当来的元素(newCome)> 堆顶元素(small ...

  8. Java中数组的几个常用算法:插入算法,删除算法,冒泡排序算法

    前言: 在Java中我们常常会用数组,提到数组就不得不介绍数组中常用到的几个算法. 有插入算法,删除算法,冒泡排序算法等. 在学习这几个数组的算法前,我们先来了解一下关于数组一些基本知识. 数组的基本 ...

  9. 【bzoj5210】最大连通子块和 树链剖分+线段树+可删除堆维护树形动态dp

    题目描述 给出一棵n个点.以1为根的有根树,点有点权.要求支持如下两种操作: M x y:将点x的点权改为y: Q x:求以x为根的子树的最大连通子块和. 其中,一棵子树的最大连通子块和指的是:该子树 ...

随机推荐

  1. mongodb数据库备份迁移 windows -> linux

    mongodb数据库备份迁移 windows -> linux cd 到本机mongodb的安装目录 如: C:\Program Files\MongoDB\Server\3.4\bin 可以发 ...

  2. 吴恩达深度学习第2课第3周编程作业 的坑(Tensorflow+Tutorial)

    可能因为Andrew Ng用的是python3,而我是python2.7的缘故,我发现了坑.如下: 在辅助文件tf_utils.py中的random_mini_batches(X, Y, mini_b ...

  3. SUSE10的虚拟机安装以及ORACLE 11g的安装

    SUSE10虚拟机安装与ORACLE安装 作者:张欣橙 本文所需要的所有参数均位于文末附录中 一.SUSE10虚拟机的安装与创建 新建虚拟机安装 选择下一步 选择下一步 选择下一步 选择下一步 选择下 ...

  4. Linux c readdir是非线程安全,需用readdir_r,要注意用静态变量当做返回值的函数的非线程安全性

    readdir函数: struct dirent *readdir(DIR *dirp); The  data  returned by readdir() may be overwritten by ...

  5. Angular4.0入门

    angular与其他的差别 angular cli安装 cnpm install -g @angular/cli 最新版本 cnpm uninstall -g @angular/cli 卸载全局版本 ...

  6. 安卓高级 Android图片缓存之初识Glide

    前言: 前面总结学习了图片的使用以及Lru算法,今天来学习一下比较优秀的图片缓存开源框架.技术本身就要不断的更迭,从最初的自己使用SoftReference实现自己的图片缓存,到后来做电商项目自己的实 ...

  7. 【Android】给Android Studio设置代理

    先打开我们的Android Studio,点击工具栏的file下的settings,如下图 之后再搜索框上面输入Proxy,然后按第四步提示点击,如下图 之后就进入了设置代理的界面了,如下图 默认情况 ...

  8. javap反编译命令详解&Eclipse中配置javap命令

    javap命令所有参数如下图所示: javap 命令用于解析类文件.其输出取决于所用的选项.若没有使用选项,javap 将输出传递给它的类的 public 域及方法.javap 将其输出到标准输出设备 ...

  9. @property的参数

    转载请标明出处: http://blog.csdn.net/xmxkf/article/details/51353580 本文出自:[openXu的博客] 参数类别 参数 说明 原子性 atomic ...

  10. [线程]Thead 中传参数RuntimeError: thread.__init__() not called

    在写一个多线程类的时候调用报错 RuntimeError: thread.__init__() not called class NotifyTread(threading.Thread): def ...