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 ...
随机推荐
- 文件的基本管理和XFS文件系统备份恢复
4.1 Linux系统目录结构和相对/绝对路径 4.1.1系统目录结构 在WIN系统中,查看文件先进入相应的盘符,然后进入文件目录 在WIN中,它是多根 c:\ d:\ e:\ Linux ...
- java--基本数据类型的转换(强制转换)
强制类型的转换 规则: 1.执行算术运算时,低类型(短字节)可以转换为高类型(长字节):例如: int型转换成double型,char型转换成int型等等. 就是用强制类型来实现. 3.强制类型转换语 ...
- css 椭圆样式
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- java基础(三):谈谈java异常的处理
1.知识点总结 1.1.异常分类 异常就是java中出现的不正常的现象(错误与异常),按照继承的体系结构,可以分类如下 Throwable: 它是所有错误与异常的超类(祖宗类) |- Error 错误 ...
- Webpack4教程 - 第二部分,使用loader处理scss,图片以及转换JS
转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者. 原文出处:https://wanago.io/2018/07/16/webpack-4-course-par ...
- 47.Odoo产品分析 (五) – 定制板块(2) – 为业务自定义odoo(2)
查看Odoo产品分析系列--目录 Odoo产品分析 (五) – 定制板块(2) – 为业务自定义odoo(1) 4 添加自定义字段 定制odoo的最普通的原因就是指定到公司的附加信息.如果您正在运行一 ...
- kotlin 第一个Android项目
一.创建过程 二.TextView点击事件 class MainActivity : AppCompatActivity() { lateinit var tv:TextView; //初始化Text ...
- vue的组件化运用(数据在两个组件互传,小问题总结)
一.vue的组件化应用 首先,知道有哪些相关的属性需要用到,再慢慢去理解,运用. 1.两个vue页面 2. slot占位符(可用可不用) 3.props内置属性 4.watch监听函数 5.impor ...
- SQL ----post漏洞测试注入
使用工具sqlmap 输入账号密码进行bp截断,获取文本保存在sqlmap下面2.txt 爆数据库 爆表爆表 爆数据 最后把数据密码md5解析
- python高级(4)—— 虚拟环境安装使用
虚拟环境 什么是虚拟环境 对电脑稍微有点常识的朋友相信都玩过,比如VMware,virtualbox,或者你用电脑端的模拟器玩手机端的游戏也是一样,其实就是一个假的空间,在Python这里,虚拟环境就 ...