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 永无乡的更多相关文章

  1. bzoj 2733 永无乡 - 并查集 - 线段树

    永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接,通过桥可以从一个岛 到达另一个岛. ...

  2. bzoj 2733 永无乡 线段树

    题目: 支持两种操作: 合并两点所在的联通块 查询某点所在联通块内权值第k小. 题解 平衡树启发式合并随便搞一搞就好了. 我写了一个线段树合并 #include <cstdio> #inc ...

  3. bzoj2733永无乡

    永无乡 HYSBZ - 2733 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接, ...

  4. BZOJ 2733: [HNOI2012]永无乡 启发式合并treap

    2733: [HNOI2012]永无乡 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...

  5. bzoj 2733: [HNOI2012]永无乡 离线+主席树

    2733: [HNOI2012]永无乡 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1167  Solved: 607[Submit][Status ...

  6. BZOJ 2733: [HNOI2012]永无乡(treap + 启发式合并 + 并查集)

    不难...treap + 启发式合并 + 并查集 搞搞就行了 --------------------------------------------------------------------- ...

  7. BZOJ 2733: [HNOI2012]永无乡 [splay启发式合并]

    2733: [HNOI2012]永无乡 题意:加边,询问一个连通块中k小值 终于写了一下splay启发式合并 本题直接splay上一个节点对应图上一个点就可以了 并查集维护连通性 合并的时候,把siz ...

  8. bzoj 2733: [HNOI2012]永无乡 -- 线段树

    2733: [HNOI2012]永无乡 Time Limit: 10 Sec  Memory Limit: 128 MB Description 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自 ...

  9. Bzoj 2733: [HNOI2012]永无乡 数组Splay+启发式合并

    2733: [HNOI2012]永无乡 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3955  Solved: 2112[Submit][Statu ...

随机推荐

  1. 文件的基本管理和XFS文件系统备份恢复

    4.1 Linux系统目录结构和相对/绝对路径 4.1.1系统目录结构 在WIN系统中,查看文件先进入相应的盘符,然后进入文件目录 在WIN中,它是多根  c:\    d:\   e:\ Linux ...

  2. java--基本数据类型的转换(强制转换)

    强制类型的转换 规则: 1.执行算术运算时,低类型(短字节)可以转换为高类型(长字节):例如: int型转换成double型,char型转换成int型等等. 就是用强制类型来实现. 3.强制类型转换语 ...

  3. css 椭圆样式

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  4. java基础(三):谈谈java异常的处理

    1.知识点总结 1.1.异常分类 异常就是java中出现的不正常的现象(错误与异常),按照继承的体系结构,可以分类如下 Throwable: 它是所有错误与异常的超类(祖宗类) |- Error 错误 ...

  5. Webpack4教程 - 第二部分,使用loader处理scss,图片以及转换JS

    转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者. 原文出处:https://wanago.io/2018/07/16/webpack-4-course-par ...

  6. 47.Odoo产品分析 (五) – 定制板块(2) – 为业务自定义odoo(2)

    查看Odoo产品分析系列--目录 Odoo产品分析 (五) – 定制板块(2) – 为业务自定义odoo(1) 4 添加自定义字段 定制odoo的最普通的原因就是指定到公司的附加信息.如果您正在运行一 ...

  7. kotlin 第一个Android项目

    一.创建过程 二.TextView点击事件 class MainActivity : AppCompatActivity() { lateinit var tv:TextView; //初始化Text ...

  8. vue的组件化运用(数据在两个组件互传,小问题总结)

    一.vue的组件化应用 首先,知道有哪些相关的属性需要用到,再慢慢去理解,运用. 1.两个vue页面 2. slot占位符(可用可不用) 3.props内置属性 4.watch监听函数 5.impor ...

  9. SQL ----post漏洞测试注入

    使用工具sqlmap 输入账号密码进行bp截断,获取文本保存在sqlmap下面2.txt 爆数据库 爆表爆表 爆数据 最后把数据密码md5解析

  10. python高级(4)—— 虚拟环境安装使用

    虚拟环境 什么是虚拟环境 对电脑稍微有点常识的朋友相信都玩过,比如VMware,virtualbox,或者你用电脑端的模拟器玩手机端的游戏也是一样,其实就是一个假的空间,在Python这里,虚拟环境就 ...