BZOJ 2733 永无乡
splay启发式合并
启发式合并其实就是把集合数量小的合并到集合数量大的里去。
怎么合并呢,直接一个一个插入就行了。。
用并查集维护连通性,find(i)可以找到所在splay的编号
这题好像还可以合并线段树来写,下次再补上。。
#include <bits/stdc++.h>
#define INF 0x3f3f3f3f
#define full(a, b) memset(a, b, sizeof a)
using namespace std;
typedef long long ll;
inline int lowbit(int x){ return x & (-x); }
inline int read(){
int X = 0, w = 0; char ch = 0;
while(!isdigit(ch)) { w |= ch == '-'; ch = getchar(); }
while(isdigit(ch)) X = (X << 3) + (X << 1) + (ch ^ 48), ch = getchar();
return w ? -X : X;
}
inline int gcd(int a, int b){ return a % b ? gcd(b, a % b) : b; }
inline int lcm(int a, int b){ return a / gcd(a, b) * b; }
template<typename T>
inline T max(T x, T y, T z){ return max(max(x, y), z); }
template<typename T>
inline T min(T x, T y, T z){ return min(min(x, y), z); }
template<typename A, typename B, typename C>
inline A fpow(A x, B p, C lyd){
A ans = 1;
for(; p; p >>= 1, x = 1LL * x * x % lyd)if(p & 1)ans = 1LL * x * ans % lyd;
return ans;
}
const int N = 100005;
int n, m, tot, fa[N<<5], ch[N<<5][2], size[N<<5], val[N<<5], root[N], parent[N], pos[N];
int find(int p){
while(p != parent[p]) parent[p] = parent[parent[p]], p = parent[p];
return p;
}
bool isConnect(int p, int q){
return find(p) == find(q);
}
int init(int v, int f){
++tot;
size[tot] = 1, val[tot] = v, fa[tot] = f;
ch[tot][0] = ch[tot][1] = 0;
return tot;
}
void push_up(int x){
size[x] = size[ch[x][0]] + size[ch[x][1]] + 1;
}
void rotate(int x){
int y = fa[x], z = fa[y], p = (ch[y][1] == x) ^ 1;
ch[y][p^1] = ch[x][p], fa[ch[x][p]] = y;
ch[z][ch[z][1] == y] = x, fa[x] = z;
ch[x][p] = y, fa[y] = x;
push_up(y), push_up(x);
}
void splay(int rt, int x, int goal){
if(x == goal) return;
while(fa[x] != goal){
int y = fa[x], z = fa[y];
if(z != goal){
(ch[y][0] == x) ^ (ch[z][0] == y) ? rotate(x) : rotate(y);
}
rotate(x);
}
push_up(x);
if(goal == 0) root[rt] = x;
}
int select(int rt, int k){
if(!root[rt]) return 0;
int cur = root[rt], p = size[ch[root[rt]][0]];
while(1){
if(p + 1 < k){
k -= p + 1;
cur = ch[cur][1];
}
else{
if(p + 1 == k) return val[cur];
cur = ch[cur][0];
}
p = size[ch[cur][0]];
}
}
void insert(int rt, int x){
if(!root[rt]){
root[rt] = init(x, 0);
return;
}
int cur = root[rt];
while(ch[cur][x > val[cur]]){
if(val[cur] == x) break;
cur = ch[cur][x > val[cur]];
}
if(val[cur] == x){
splay(rt, cur, 0);
return;
}
ch[cur][x > val[cur]] = init(x, cur);
splay(rt, ch[cur][x > val[cur]], 0);
}
void dfs(int x, int y){
if(!x) return;
dfs(ch[x][0], y);
dfs(ch[x][1], y);
insert(y, val[x]);
}
void merge(int x, int y){
int fx = find(x), fy = find(y);
if(fx == fy) return;
if(size[root[fx]] > size[root[fy]]) swap(fx, fy);
parent[fx] = fy;
dfs(root[fx], fy);
}
int main(){
n = read(), m = read();
for(int i = 0; i <= n; i ++) parent[i] = i;
for(int i = 1; i <= n; i ++){
int v = read();
root[i] = init(v, 0);
pos[v] = i;
}
for(int i = 0; i < m; i ++){
int x = read(), y = read();
if(isConnect(x, y)) continue;
merge(x, y);
}
int q = read();
while(q --){
char opt[5]; scanf("%s", opt);
int x = read(), y = read();
if(opt[0] == 'B'){
if(isConnect(x, y)) continue;
merge(x, y);
}
else if(opt[0] == 'Q'){
if(size[root[find(x)]] < y) printf("-1\n");
else printf("%d\n", pos[select(find(x), y)]);
}
}
return 0;
}
写个下动态开点的权值线段树,为啥跑出来比splay还要慢啊!这么玄学的吗。。
#include <bits/stdc++.h>
#define INF 0x3f3f3f3f
#define full(a, b) memset(a, b, sizeof a)
using namespace std;
typedef long long ll;
inline int lowbit(int x){ return x & (-x); }
inline int read(){
int X = 0, w = 0; char ch = 0;
while(!isdigit(ch)) { w |= ch == '-'; ch = getchar(); }
while(isdigit(ch)) X = (X << 3) + (X << 1) + (ch ^ 48), ch = getchar();
return w ? -X : X;
}
inline int gcd(int a, int b){ return a % b ? gcd(b, a % b) : b; }
inline int lcm(int a, int b){ return a / gcd(a, b) * b; }
template<typename T>
inline T max(T x, T y, T z){ return max(max(x, y), z); }
template<typename T>
inline T min(T x, T y, T z){ return min(min(x, y), z); }
template<typename A, typename B, typename C>
inline A fpow(A x, B p, C lyd){
A ans = 1;
for(; p; p >>= 1, x = 1LL * x * x % lyd)if(p & 1)ans = 1LL * x * ans % lyd;
return ans;
}
const int N = 100005;
int n, m, tot;
int tree[N<<5], root[N], lc[N<<5], rc[N<<5], pos[N], parent[N];
int find(int p){
while(p != parent[p]) parent[p] = parent[parent[p]], p = parent[p];
return p;
}
int build(){
++tot;
tree[tot] = lc[tot] = rc[tot] = 0;
return tot;
}
void push_up(int x){
tree[x] = tree[lc[x]] + tree[rc[x]];
}
void modify(int rt, int l, int r, int k){
if(l == r){
tree[rt] ++;
return;
}
int mid = (l + r) >> 1;
if(k <= mid){
if(!lc[rt]) lc[rt] = build();
modify(lc[rt], l, mid, k);
}
else{
if(!rc[rt]) rc[rt] = build();
modify(rc[rt], mid + 1, r, k);
}
push_up(rt);
}
int merge(int x, int y, int l, int r){
if(!x) return y;
if(!y) return x;
if(l == r){
tree[x] += tree[y];
return x;
}
int mid = (l + r) >> 1;
lc[x] = merge(lc[x], lc[y], l, mid);
rc[x] = merge(rc[x], rc[y], mid + 1, r);
push_up(x);
return x;
}
int query(int rt, int l, int r, int k){
if(l == r) return l;
int mid = (l + r) >> 1;
if(k <= tree[lc[rt]]) return query(lc[rt], l, mid, k);
return query(rc[rt], mid + 1, r, k - tree[lc[rt]]);
}
int main(){
n = read(), m = read();
for(int i = 0; i <= n; i ++) parent[i] = i;
for(int i = 1; i <= n; i ++){
int v = read();
pos[v] = i, root[i] = build(), modify(root[i], 1, n, v);
}
for(int i = 0; i < m; i ++){
int x = read(), y = read();
int fx = find(x), fy = find(y);
if(fx == fy) continue;
parent[fy] = fx;
root[fx] = merge(root[fx], root[fy], 1, n);
}
int q = read();
while(q --){
char opt[10]; scanf("%s", opt);
if(opt[0] == 'B'){
int x = read(), y = read();
int fx = find(x), fy = find(y);
if(fx == fy) continue;
parent[fy] = fx;
root[fx] = merge(root[fx], root[fy], 1, n);
}
else if(opt[0] == 'Q'){
int x = read(), k = read();
if(tree[root[find(x)]] < k) printf("-1\n");
else printf("%d\n", pos[query(root[find(x)], 1, n, k)]);
}
}
return 0;
}
BZOJ 2733 永无乡的更多相关文章
- bzoj 2733 永无乡 - 并查集 - 线段树
永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接,通过桥可以从一个岛 到达另一个岛. ...
- bzoj 2733 永无乡 线段树
题目: 支持两种操作: 合并两点所在的联通块 查询某点所在联通块内权值第k小. 题解 平衡树启发式合并随便搞一搞就好了. 我写了一个线段树合并 #include <cstdio> #inc ...
- bzoj2733永无乡
永无乡 HYSBZ - 2733 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接, ...
- BZOJ 2733: [HNOI2012]永无乡 启发式合并treap
2733: [HNOI2012]永无乡 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...
- bzoj 2733: [HNOI2012]永无乡 离线+主席树
2733: [HNOI2012]永无乡 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1167 Solved: 607[Submit][Status ...
- BZOJ 2733: [HNOI2012]永无乡(treap + 启发式合并 + 并查集)
不难...treap + 启发式合并 + 并查集 搞搞就行了 --------------------------------------------------------------------- ...
- BZOJ 2733: [HNOI2012]永无乡 [splay启发式合并]
2733: [HNOI2012]永无乡 题意:加边,询问一个连通块中k小值 终于写了一下splay启发式合并 本题直接splay上一个节点对应图上一个点就可以了 并查集维护连通性 合并的时候,把siz ...
- bzoj 2733: [HNOI2012]永无乡 -- 线段树
2733: [HNOI2012]永无乡 Time Limit: 10 Sec Memory Limit: 128 MB Description 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自 ...
- Bzoj 2733: [HNOI2012]永无乡 数组Splay+启发式合并
2733: [HNOI2012]永无乡 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 3955 Solved: 2112[Submit][Statu ...
随机推荐
- c++性能测试工具:google benchmark入门(一)
如果你正在寻找一款c++性能测试工具,那么这篇文章是不容错过的. 市面上的benchmark工具或多或少存在一些使用上的不便,那么是否存在一个使用简便又功能强大的性能测试工具呢?答案是google/b ...
- 杭电ACM2022--海选女主角
海选女主角 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Submi ...
- c# Base64解密加密
private static string base64EncodeChars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz ...
- 常用vi编辑器命令行
游标控制: h 游标向左移 j 游标向下移 k 游标向上移 l(or spacebar) 游标向右移 w 向前移动一个单词 b 向后移动一个单词 e 向前移动一个单词,且游标指向单词的末尾 ( 移到当 ...
- java中求质数(素数)的问题
这篇笔记讲讲关于java中质数的问题. 一.什么是质数(素数)? 定义:质数又称素数.一个大于1的自然数,除了1和它自身外,不能被其他自然数整除的数叫做质数:否则称为合数.它可以有无限个数. 二.ja ...
- C++系列总结——封装
前言 众所周知,封装.继承和多态是面向对象编程的三大特性.C++作为一门面向对象的编程语言,自然支持了这些特性,但C++是如何实现这些特性的呢?今天先说下我理解的封装. 封装 通常我们会把下面的行为也 ...
- 关于获取URL中传值的解决方法--升级版
这次页面之间的传值是升级版本,为什么是升级版本呢,因为这次页面的传值不一样了.大家可以看一下我原来的文章<关于获取URL中传值的解决方法> 其实上次就已经比较清楚的介绍了页面之间的传值,但 ...
- Chart.js 與 ASP.NET MVC 整合應用
Chart.js 是一套開放原始碼的「圖表」繪製函式庫,和其他第三方的圖表工具相比,Chart.js 的特色如下: 支援 HTML 5.響應式網頁 (RWD, Responsive Web Desig ...
- 【English】七、常见动词
一.动词: touch.hear.say.listen touch [tʌtʃ] 触摸 I touch the cat. They touch the elephant. hear [hɪr] 听到 ...
- js 判断元素(例如div)里的数据显示不全(数据长度大于元素长度)
//判断div里元素是否超出长度,true 超出,false 没有 dom=document.getElementById('id');function isEllipsis(dom) { va ...