题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1512
题意: 有n个猴子,一开始每个猴子只认识自己。每个猴子有一个力量值,力量值越大表示这个猴子打架越厉害。如果2个猴子不认识,他们就会找他们认识的猴子中力量最大的出来单挑,单挑不论输赢,单挑的2个猴子力量值减半,这2拨猴子就都认识了,不打不相识嘛。现在给m组询问,如果2只猴子相互认识,输出-1,否则他们各自找自己认识的最牛叉的猴子单挑,求挑完后这拨猴子力量最大值。
解题方法:首先很明显这题涉及到集合并的操作,要用到并查集。其次要找到某一拨猴子中力量最大值,找最大值最快的应该是堆。2拨猴子要快速合并而又不失堆的特性,想来想去左偏树比较合适。下面来说说左偏树:
然后左偏树有几个很重要的性质如下:
[性质1] 节点的键值小于或等于它的左右子节点的键值。
[性质2] 节点的左子节点的距离不小于右子节点的距离。
[性质3] 节点的左子节点右子节点也是一颗左偏树。
左偏树支持,插入,删除,去最小节点,以及最重要的合并操作,并且保证这些操作的复杂度为严格的log(n)。
下面给出合并操作的核心代码:
int merge(int a, int b) //合并
{
if(a == 0) return b;
if(b == 0) return a;
if(heap[a].v < heap[b].v) swap(a, b);
heap[a].r = merge(heap[a].r, b);
heap[heap[a].r].f = a;
if(heap[heap[a].l].dis < heap[heap[a].r].dis) swap(heap[a].l, heap[a].r);
if(heap[a].r == 0) heap[a].dis = 0;
else heap[a].dis = heap[heap[a].r].dis + 1;
return a;
}
然后给出本题的完整代码,第一次写可并堆,代码是参考了别人写的。
//hdoj 1512 可并堆
#include <bits/stdc++.h>
using namespace std;
const int maxn = 100005;
struct tree{int l, r, v, dis, f;}heap[maxn]; //左儿子,右儿子,权值, 距离,父亲节点
int merge(int a, int b) //合并
{
if(a == 0) return b;
if(b == 0) return a;
if(heap[a].v < heap[b].v) swap(a, b);
heap[a].r = merge(heap[a].r, b);
heap[heap[a].r].f = a;
if(heap[heap[a].l].dis < heap[heap[a].r].dis) swap(heap[a].l, heap[a].r);
if(heap[a].r == 0) heap[a].dis = 0;
else heap[a].dis = heap[heap[a].r].dis + 1;
return a;
}
int pop(int a){ //删除
int l = heap[a].l, r = heap[a].r;
heap[l].f = l, heap[r].f = r;
heap[a].l = heap[a].r = heap[a].dis = 0;
return merge(l, r);
}
int find(int a){ //查找
return heap[a].f == a ? a : find(heap[a].f);
}
int main(){
int n, m;
while(scanf("%d", &n) != EOF)
{
for(int i = 1; i <= n; i++){
scanf("%d", &heap[i].v);
heap[i].l = heap[i].r = heap[i].dis = 0;
heap[i].f = i;
}
scanf("%d", &m);
for(int i = 1; i <= m; i++){
int a, b, fa, fb;
scanf("%d%d", &a, &b);
fa = find(a), fb = find(b);
if(fa == fb){
printf("-1\n");
}
else{
heap[fa].v /= 2;
int u = pop(fa);
u = merge(u, fa);
heap[fb].v /= 2;
int v = pop(fb);
v = merge(v, fb);
printf("%d\n", heap[merge(u, v)].v);
}
}
}
return 0;
}