Blog of RuSun

\begin {array}{c} \mathfrak {One Problem Is Difficult} \\\\ \mathfrak {Because You Don't Know} \\\\ \mathfrak {Why It Is Diffucult} \end {array}

左偏树模板

用于可并堆。

左偏树不能保证树的深度,所以用并查集维护堆的集合。

查看代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
#include <cstdio>
#include <algorithm>
using namespace std;
const int N = 1e5 + 10;
int n, m, p[N];
struct Node
{
bool vis;
int id, v, d, ls, rs;
bool operator>(Node &x) const
{
return v > x.v || (v == x.v && id > x.id);
}
} tr[N];
int fa(int x)
{
return x == p[x] ? x : p[x] = fa(p[x]);
}
int merge(int x, int y)
{
if (!x || !y)
return x | y;
if (tr[x] > tr[y])
swap(x, y);
tr[x].rs = merge(tr[x].rs, y);
if (tr[tr[x].rs].d > tr[tr[x].ls].d)
swap(tr[x].ls, tr[x].rs);
tr[x].d = tr[tr[x].rs].d + 1;
return x;
}
void remove(int x)
{
tr[x].vis = true;
p[tr[x].ls] = p[tr[x].rs] = p[x] = merge(tr[x].ls, tr[x].rs);
tr[x].ls = tr[x].rs = tr[x].d = 0;
}
int main()
{
scanf("%d%d", &n, &m);
for (int i = 1; i <= n; i++)
{
scanf("%d", &tr[i].v);
p[i] = tr[i].id = i;
}
for (int op; m; m--)
{
scanf("%d", &op);
if (op == 1)
{
int x, y;
scanf("%d%d", &x, &y);
if (tr[x].vis || tr[x].vis)
continue;
x = fa(x);
y = fa(y);
if (x != y)
p[x] = p[y] = merge(x, y);
}
if (op == 2)
{
int x;
scanf("%d", &x);
if (tr[x].vis)
{
puts("-1");
continue;
}
x = fa(x);
printf("%d\n", tr[x].v);
remove(x);
}
}
return 0;
}