mirror of
https://git.sb/baoshuo/OI-codes.git
synced 2025-01-23 13:31:58 +00:00
2733. [HNOI2012] 永无乡
https://hydro.ac/d/bzoj/record/639990b17e5c9216cb39a349
This commit is contained in:
parent
3ed4a1e9ac
commit
c1ebbefb17
228
BZOJ/2733/2733.cpp
Normal file
228
BZOJ/2733/2733.cpp
Normal file
@ -0,0 +1,228 @@
|
||||
#include <iostream>
|
||||
#include <cstdlib>
|
||||
#include <limits>
|
||||
#include <numeric>
|
||||
|
||||
using std::cin;
|
||||
using std::cout;
|
||||
const char endl = '\n';
|
||||
|
||||
const int N = 1e5 + 5;
|
||||
|
||||
class Treap {
|
||||
private:
|
||||
struct node {
|
||||
node *left, *right;
|
||||
int size, val, key;
|
||||
|
||||
node(const int &_val = 0)
|
||||
: left(nullptr), right(nullptr), size(1), val(_val), key(rand()) {}
|
||||
|
||||
~node() {
|
||||
delete left;
|
||||
delete right;
|
||||
}
|
||||
|
||||
void pushup() {
|
||||
size = 1;
|
||||
|
||||
if (left != nullptr) size += left->size;
|
||||
if (right != nullptr) size += right->size;
|
||||
}
|
||||
} * root;
|
||||
|
||||
inline int getNodeSize(node *node) {
|
||||
return node == nullptr ? 0 : node->size;
|
||||
}
|
||||
|
||||
node *find(node *p, int val) {
|
||||
if (p == nullptr) return nullptr;
|
||||
if (p->val == val) return p;
|
||||
if (p->val > val) return find(p->left, val);
|
||||
return find(p->right, val);
|
||||
}
|
||||
|
||||
std::pair<node *, node *> split(node *p, int k) {
|
||||
if (p == nullptr) return std::make_pair(nullptr, nullptr);
|
||||
|
||||
std::pair<node *, node *> o;
|
||||
|
||||
if (k <= getNodeSize(p->left)) {
|
||||
o = split(p->left, k);
|
||||
p->left = o.second;
|
||||
p->pushup();
|
||||
o.second = p;
|
||||
} else {
|
||||
o = split(p->right, k - getNodeSize(p->left) - 1);
|
||||
p->right = o.first;
|
||||
p->pushup();
|
||||
o.first = p;
|
||||
}
|
||||
|
||||
return o;
|
||||
}
|
||||
|
||||
std::pair<node *, node *> splitByValue(node *p, int val) {
|
||||
if (p == nullptr) return std::make_pair(nullptr, nullptr);
|
||||
|
||||
std::pair<node *, node *> o;
|
||||
|
||||
if (p->val < val) {
|
||||
o = splitByValue(p->right, val);
|
||||
p->right = o.first;
|
||||
p->pushup();
|
||||
o.first = p;
|
||||
} else {
|
||||
o = splitByValue(p->left, val);
|
||||
p->left = o.second;
|
||||
p->pushup();
|
||||
o.second = p;
|
||||
}
|
||||
|
||||
return o;
|
||||
}
|
||||
|
||||
node *merge(node *x, node *y) {
|
||||
if (x == nullptr) return y;
|
||||
if (y == nullptr) return x;
|
||||
|
||||
if (x->key > y->key) {
|
||||
x->right = merge(x->right, y);
|
||||
x->pushup();
|
||||
return x;
|
||||
}
|
||||
|
||||
y->left = merge(x, y->left);
|
||||
y->pushup();
|
||||
|
||||
return y;
|
||||
}
|
||||
|
||||
void _merge(node *cur) {
|
||||
if (cur->left != nullptr) _merge(cur->left);
|
||||
if (cur->right != nullptr) _merge(cur->right);
|
||||
|
||||
insert(cur->val);
|
||||
}
|
||||
|
||||
public:
|
||||
Treap()
|
||||
: root(nullptr) {}
|
||||
|
||||
~Treap() {
|
||||
delete root;
|
||||
}
|
||||
|
||||
void insert(int val) {
|
||||
auto o = splitByValue(root, val);
|
||||
o.first = merge(o.first, new Treap::node(val));
|
||||
root = merge(o.first, o.second);
|
||||
}
|
||||
|
||||
void erase(int val) {
|
||||
auto o = splitByValue(root, val);
|
||||
auto t = o;
|
||||
|
||||
if (find(o.second, val) != nullptr) {
|
||||
t = split(o.second, 1);
|
||||
delete t.first;
|
||||
}
|
||||
|
||||
root = merge(o.first, t.second);
|
||||
}
|
||||
|
||||
int getRank(int val) {
|
||||
auto x = splitByValue(root, val);
|
||||
int r = getNodeSize(x.first) + 1;
|
||||
|
||||
root = merge(x.first, x.second);
|
||||
|
||||
return r;
|
||||
}
|
||||
|
||||
int getKth(int k) {
|
||||
auto x = split(root, k - 1);
|
||||
auto y = split(x.second, 1);
|
||||
|
||||
Treap::node *o = y.first;
|
||||
|
||||
root = merge(x.first, merge(y.first, y.second));
|
||||
|
||||
return o == nullptr ? 0 : o->val;
|
||||
}
|
||||
|
||||
int size() {
|
||||
return getNodeSize(root);
|
||||
}
|
||||
|
||||
void merge(Treap &b) {
|
||||
_merge(b.root);
|
||||
}
|
||||
} tr[N];
|
||||
|
||||
int n, m, q, a[N], fa[N];
|
||||
|
||||
int find(int x) {
|
||||
return fa[x] == x ? x : fa[x] = find(fa[x]);
|
||||
}
|
||||
|
||||
void merge(int x, int y) {
|
||||
x = find(x);
|
||||
y = find(y);
|
||||
|
||||
if (x != y) {
|
||||
if (tr[x].size() < tr[y].size()) {
|
||||
fa[x] = y;
|
||||
tr[y].merge(tr[x]);
|
||||
} else {
|
||||
fa[y] = x;
|
||||
tr[x].merge(tr[y]);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int main() {
|
||||
std::ios::sync_with_stdio(false);
|
||||
cin.tie(nullptr);
|
||||
|
||||
cin >> n >> m;
|
||||
|
||||
a[0] = -1;
|
||||
|
||||
for (int i = 1, x; i <= n; i++) {
|
||||
cin >> x;
|
||||
|
||||
a[x] = i;
|
||||
fa[i] = i;
|
||||
tr[i].insert(x);
|
||||
}
|
||||
|
||||
for (int i = 1, u, v; i <= m; i++) {
|
||||
cin >> u >> v;
|
||||
|
||||
merge(u, v);
|
||||
}
|
||||
|
||||
cin >> q;
|
||||
|
||||
while (q--) {
|
||||
char op;
|
||||
int x, y;
|
||||
|
||||
cin >> op >> x >> y;
|
||||
|
||||
if (op == 'Q') {
|
||||
auto &t = tr[find(x)];
|
||||
|
||||
if (t.size() < y) {
|
||||
cout << -1 << endl;
|
||||
} else {
|
||||
cout << a[t.getKth(y)] << endl;
|
||||
}
|
||||
} else { // op == 'B'
|
||||
merge(x, y);
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
BIN
BZOJ/2733/data/1.in
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/1.in
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/1.out
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/1.out
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/10.in
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/10.in
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/10.out
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/10.out
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/2.in
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/2.in
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/2.out
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/2.out
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/3.in
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/3.in
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/3.out
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/3.out
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/4.in
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/4.in
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/4.out
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/4.out
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/5.in
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/5.in
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/5.out
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/5.out
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/6.in
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/6.in
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/6.out
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/6.out
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/7.in
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/7.in
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/7.out
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/7.out
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/8.in
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/8.in
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/8.out
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/8.out
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/9.in
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/9.in
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
BZOJ/2733/data/9.out
(Stored with Git LFS)
Normal file
BIN
BZOJ/2733/data/9.out
(Stored with Git LFS)
Normal file
Binary file not shown.
Loading…
Reference in New Issue
Block a user