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. cocos creator主程入门教程(九)—— 瓦片地图

    五邑隐侠,本名关健昌,10年游戏生涯,现隐居五邑.本系列文章以TypeScript为介绍语言. 这一篇介绍瓦片地图,在开发模拟经营类游戏.SLG类游戏.RPG游戏,都会使用到瓦片地图.瓦片地图地面是通 ...

  2. golang实现aes-cbc-256加密解密过程记录

    我为什么吃撑了要实现go的aes-cbc-256加密解密功能? 之前的项目是用php实现的,现在准备用go重构,需要用到这个功能,这么常用的功能上网一搜一大把现成例子,于是基于go现有api分分钟实现 ...

  3. [TCP/IP] 网络层-抓包分析IP数据包首部

    ip数据包的结构:首部+数据部分 1.版本(v4或者v6)+首部长度(固定的20字节,所以就没有)+区分服务优先级(我的例子是 assured forwarding 31 0x1a 26,保证转发) ...

  4. 学代码第十七天,JAVA继承

    JANA面向对象的三大特性:封装,继承,多态. 今天学了继承,继承,通俗点说就是子类可以用父类的代码,或重写父类的方法.构造方法.属性 例如我这里要调用父类的方法: 下边有两个测试类,自己分别试一下, ...

  5. 人脸检测识别,人脸检测,人脸识别,离线检测,C#源码

    百度网盘地址 微云地址 使用虹软人工智能开放平台技术开发完成

  6. Tableau环图可视化

    1.选择"记录数",拖拽两个记录数放入列中,求总和,选择饼图: 2.选择"大小",调整两个饼图的大小: 3.点击第二个总和(行上的),选择“双轴”: 4.点击坐 ...

  7. K邻近回归算法

    代码: # -*- coding: utf-8 -*- """ Created on Fri Jul 13 10:40:22 2018 @author: zhen &qu ...

  8. ASP.NET Zero--单元测试

    单元测试 ASP.NET Zero启动项目包含单元和集成测试.使用以下工具开发测试: xUnit作为测试框架. Shouldly 作为断言库. Microsoft.EntityFrameworkCor ...

  9. Hibernate执行SQL语句实现查询修改功能!

    今天玩Hibernate时突然就想写写SQL语句查询... DAO : //查询 public List<?> createSqlQueryList(final String queryS ...

  10. Java基础之入门

    写写基础,顺便回顾下,再深层次思考下哪些深入的没弄明白. Java是Sun Microsystems于1995年推出的高级编程语言  其版本 由 1.1 -> 1.2 -> 1.3 -&g ...