本题在上一题基础上多加了一个heap_swap映射,还有up函数
#include<iostream>
#include<algorithm>
using namespace std;
const int N=100010;
int n,m=0;
int h[N],ph[N],hp[N],cnt;//h代表heap(堆),ph(point->heap)可以获得第几个插入的元素现在在堆的哪个位置
//hp(heap->point)可以获得在堆的第n个元素存的是第几个插入的元素
void heap_swap(int a,int b){//交换在heap中位置分别为a,b的两个元素
swap(ph[hp[a]],ph[hp[b]]);//根据a和b的位置找到它们分别是第几个插入的元素,然后将其(在h数组中的)下标转换
swap(hp[a],hp[b]);//将两个位置存的是第几号元素转换
swap(h[a],h[b]);//最后再转换值(这三个语句位置可以换,但是从上到下逐渐变短的话比较美观)
}
void down(int u){//当前堆的元素下沉
int t=u;
if(u*2<=cnt&&h[u*2]<h[t])t=u*2;
if(u*2+1<=cnt&&h[u*2+1]<h[t])t=u*2+1;
if(u!=t){
heap_swap(u,t);
down(t);
}
}
void up(int u){
while(u/2 and h[u/2]>h[u]){//第一个u/2是防止当u冲到顶然后陷入死循环
heap_swap(u/2,u);
u/=2;
}
}
int main(){
cin>>n;
while(n--){
string op;
int k,x;
cin>>op;
if(op=="I"){//插入
cin>>x;
cnt++,m++;//初始化
ph[m]=cnt;//m代表是第几个插入的元素(point)->cnt指向的是插入的位置(heap)
hp[cnt]=m;//原理同上
h[cnt]=x;
up(cnt);
}
else if(op=="PM"){//输出最小
cout<<h[1]<<endl;
}
else if(op=="DM"){//删除最小
heap_swap(1,cnt);
cnt--;
down(1);
}
else if(op=="D"){//删除第k个插入的数
cin>>k;
k=ph[k];//k从储存第几个输入的数字变换为储存那个数字存放在h的哪个位置
heap_swap(k,cnt);
cnt--;
down(k);//其可能大,可能小,都操作一遍准没错
up(k);
}
else{//修改第k个插入的数
cin>>k>>x;
k=ph[k];//k从储存第几个输入的数字变换为储存那个数字存放在h的哪个位置
h[k]=x;
down(k);
up(k);
}
}
return 0;
}