堆 + 模板题---数组模拟堆

题目描述

维护一个集合,初始时集合为空,支持如下几种操作:
“I x”,插入一个数x;
“PM”,输出当前集合中的最小值;
“DM”,删除当前集合中的最小值(数据保证此时的最小值唯一);
“D k”,删除第k个插入的数;
“C k x”,修改第k个插入的数,将其变为x;
现在要进行N次操作,对于所有第2个操作,输出当前集合的最小值。
输入格式
第一行包含整数N。
接下来N行,每行包含一个操作指令,操作指令为”I x”,”PM”,”DM”,”D k”或”C k x”中的一种。
输出格式
对于每个输出指令“PM”,输出一个结果,表示当前集合中的最小值。
每个结果占一行。
数据范围
1≤N≤105
−109≤x≤109
数据保证合法。
输入样例:
8
I -10
PM
I -10
D 1
C 2 8
I 6
PM
DM
输出样例:
-10
6

题解

这里要注意的是heap_swap() 这个函数。
现在考虑要交换堆中的两个数3和5,其中3是第k个插入的数,在堆中的位置是size, 5是第m个插入的数,在堆中的位置是size/2.即hp[size] = k, ph[k] = size; hp[size/2]=m, ph[m] = size;
我们要交换3和5,不仅要交换这两个值,还要让他们的插入顺序这一信息跟着变动(如果不交换各自的插入顺序,那么5就变成第k个,3变成第m个)。那么,交换之后的结果应该是在堆中位置size存放5,且hp[size] = m,ph[m] = size; 同样在堆中位置size/2存放的是3,且hp[size/2] = k, ph[k] = size/2;
可以看到hp[], ph[]数组的值都要相互交换,于是就有了heap_swap()函数里面的操作:

代码

#include <iostream>
#include <string>
using namespace std;

const int N = 100010;
//ph[k] = m数组表示第k个插入的元素在数组中下标是m,
//hp[k] = m,表示数组下标是k的元素时第m次插入的
int h[N], ph[N], hp[N], cur_size; 

void heap_swap(int a, int b) //参数是数组下标
{
    //交换ph数组,这里要用到hp数组, 因为传入的参数是数组下标,而交换ph数组需要知道数组下标对应的第几个插入的,用hp数组查找
    swap(ph[hp[a]], ph[hp[b]]);
    
    //交换hp数组
    swap(hp[a], hp[b]);
    
    //交换值
    swap(h[a], h[b]);

}

void down(int u)
{
    int t = u;
    //改变u的指向
    if(2 * u <= cur_size && h[2 * u] < h[t]) t = 2 * u;
    if(2 * u + 1 <= cur_size && h[2 * u + 1] < h[t]) t = 2 * u + 1;
    if(u != t) //t有改变
    {
        heap_swap(u, t);
        down(t);
    }
}

void up(int u)
{
    while(u / 2 && h[u / 2] > h[u]) //h数组下标从1开始,u / 2 是 u 的父节点
    {
        heap_swap(u, u / 2);
        u >>= 1; //改变u的指向
    }
}

int main()
{
    int n;
    cin >> n;
    int m = 0; //记录第几个插入
    while(n--)
    {
        string op;
        int k, x;
        cin >> op;
        if(op=="I")
        {
            cin>>x;
            m++;
            h[++cur_size]=x;
            ph[m]=cur_size;
            hp[cur_size]=m;
            //down(size);
            up(cur_size);
        }
        else if(op=="PM")    cout<<h[1]<<endl;
        else if(op=="DM")
        {
            heap_swap(1,cur_size);
            cur_size--;
            down(1);
        }
        
        else if(op=="D")
        {
            cin>>k;
            int u=ph[k];                //这里一定要用u=ph[k]保存第k个插入点的下标
            heap_swap(u,cur_size);          //因为在此处heap_swap操作后ph[k]的值已经发生 
            cur_size--;                    //如果在up,down操作中仍然使用ph[k]作为参数就会发生错误
            up(u);
           down(u);
        }
 
        else if(op=="C")
        {
            cin>>k>>x;
            h[ph[k]]=x;                 //此处由于未涉及heap_swap操作且下面的up、down操作只会发生一个所以
            down(ph[k]);                //所以可直接传入ph[k]作为参数
            up(ph[k]);
        }
    }
    
    return 0;
}



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值