题解【bzoj2733 [HNOI2012]永无乡】
Descriprition
两种操作
- 把两个集合并起来
- 求一个集合中的第 \(k\) 大(的编号)
\(n \leq 10^5\)
Solution
平衡树的板子题之一
维护两个点连不连通直接并查集
考虑怎么把两个集合合并
启发式合并!即把 siz 小的那一颗平衡树每一个点暴力地加入到另一个
这样做的复杂度?对于每一个点,每一次合并之后集合大小都至少是原来的两边,所以每一个点都只会被合并 \(\log n\) 次。所以这样做是 \(O(n \log n)\) 的。
实现上的细节问题:
我用了 fhqtreap(大法好!)。启发式合并的过程(借鉴了题解区里另外一个dalao的fhqtreap)可以这么写:
inline void M(node *&r, node *p) { // p 合并到 r 中
if(!p) return ;
M(r, p->ch[0]); M(r, p->ch[1]); // 递归左子树和右子树
p->ch[0] = p->ch[1] = 0; // 把它左右子清空然后插到 r 里
insert(r, p);
}
Code
#include <bits/stdc++.h>
using namespace std;
const int N = 100100;
int n, m, fa[N];
struct node {
int d, id, siz, rnd;
node *ch[2];
inline void upd() {
int ret = 1;
if(ch[0]) ret += ch[0]->siz;
if(ch[1]) ret += ch[1]->siz;
siz = ret;
}
}pool[N], *cur = pool, *root[N];
inline int find(int x) {
return x == fa[x] ? x : fa[x] = find(fa[x]);
}
inline int siz(node *p) {
if(p) return p->siz; return 0;
}
inline node *newnode(int d, int id) {
node *p = cur++; p->rnd = (rand() << 15) + rand();
p->siz = 1, p->d = d; p->id = id;
p->ch[0] = p->ch[1] = 0;
return p;
}
inline node *merge(node *p, node *q) {
if(!p) return q;
if(!q) return p;
if(p->rnd < q->rnd) { p->ch[1] = merge(p->ch[1], q); p->upd(); return p; }
if(p->rnd >= q->rnd) { q->ch[0] = merge(p, q->ch[0]); q->upd(); return q; }
}
inline void split(node *r, int k, node *&p, node *&q) {
if(!r) { p = q = 0; return ; }
if(siz(r->ch[0]) < k) p = r, split(r->ch[1], k - siz(r->ch[0]) - 1, r->ch[1], q);
else q = r, split(r->ch[0], k, p, r->ch[0]); r->upd();
}
inline int rk(node *r, int x) {
if(!r) return 0;
if(r->d >= x) return rk(r->ch[0], x);
else return rk(r->ch[1], x) + siz(r->ch[0]) + 1;
}
inline void insert(node *&r, node *x) {
node *p, *q; int k = rk(r, x->d);
split(r, k, p, q);
r = merge(merge(p, x), q);
}
inline void M(node *&r, node *p) { // p -> r
if(!p) return ;
M(r, p->ch[0]); M(r, p->ch[1]);
p->ch[0] = p->ch[1] = 0; p->upd(); insert(r, p);
}
inline node *Merge(node *p, node *q) {
if(p->siz <= q->siz) swap(p, q); M(p, q); return p;
}
/*
inline void out(node *p) {
if(p->ch[0]) out(p->ch[0]);
printf("%d ", p->d);
if(p->ch[1]) out(p->ch[1]);
}
*/
int main() {
scanf("%d %d", &n, &m);
for(int i = 1; i <= n; i++) fa[i] = i;
for(int i = 1; i <= n; i++) {
int x; scanf("%d", &x);
root[i] = newnode(x, i);
} int q;
for(int i = 1; i <= m; i++) {
int u, v; scanf("%d %d", &u, &v);
int fu = find(u), fv = find(v);
if(fu == fv) continue ;
root[fu] = Merge(root[fu], root[fv]);
fa[fv] = fu;
}
scanf("%d", &q);
for(int i = 1; i <= q; i++) {
char op[5]; int x, y;
scanf("%s %d %d", op, &x, &y);
if(op[0] == 'B') {
int fx = find(x), fy = find(y);
if(fx == fy) continue ;
root[fx] = Merge(root[fx], root[fy]);
fa[fy] = fx;
} else {
int fx = find(x);
node *p, *q, *r;
if(siz(root[fx]) < y) {
printf("-1\n"); continue ;
}
split(root[fx], y - 1, p, q);
split(q, 1, q, r);
printf("%d\n", q->id);
root[fx] = merge(p, merge(q, r));
}
}
return 0;
}
题解【bzoj2733 [HNOI2012]永无乡】的更多相关文章
- bzoj2733: [HNOI2012]永无乡 启发式合并
地址:http://www.lydsy.com/JudgeOnline/problem.php?id=2733 题目: 2733: [HNOI2012]永无乡 Time Limit: 10 Sec ...
- bzoj2733: [HNOI2012]永无乡(splay)
2733: [HNOI2012]永无乡 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 3778 Solved: 2020 Description 永 ...
- [Bzoj2733][Hnoi2012] 永无乡(BST)(Pb_ds tree)
2733: [HNOI2012]永无乡 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 4108 Solved: 2195[Submit][Statu ...
- [bzoj2733][HNOI2012]永无乡_权值线段树_线段树合并
永无乡 bzoj-2733 HNOI-2012 题目大意:题目链接. 注释:略. 想法: 它的查询操作非常友善,就是一个联通块内的$k$小值. 故此我们可以考虑每个联通块建一棵权值线段树. 这样的话每 ...
- bzoj2733: [HNOI2012]永无乡 线段树合并
永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接,通过桥可以从一个岛 到达另一个岛. ...
- BZOJ2733 [HNOI2012]永无乡 【线段树合并】
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...
- [BZOJ2733] [HNOI2012] 永无乡 (splay启发式合并)
Description 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接,通过桥可以 ...
- BZOJ2733[HNOI2012]永无乡——线段树合并+并查集+启发式合并
题目描述 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接,通过桥可以从一个岛 到达 ...
- BZOJ2733: [HNOI2012]永无乡(线段树合并)
Description 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接,通过桥可以 ...
随机推荐
- bc命令详解
基础命令学习目录首页 原文链接:https://www.cnblogs.com/lovevivi/p/4359296.html 最近经常要在linux下做一些进制转换,看到了可以使用bc命令,如下: ...
- 9.Hive Metastore Administration
前言metastore参数metastore的基本参数metastore的额外参数客户端参数使用zk自动发现mestastore启动hive metastore服务 前言 本节讲metastore相关 ...
- #1490 : Tree Restoration-(微软2017在线笔试)
输入n m km个数,表示每层的节点个数接下来m行是每层的节点,节点顺序是从左往右的k个叶子节点k*k个矩阵,表示叶子节点之间的距离 输出:每个节点的父亲节点编号,root节点是0 题解:1.很明显, ...
- 奔跑吧DKY——团队Scrum冲刺阶段-Day 1-领航
各个成员在 Alpha 阶段认领的任务 修改 序号 修改 具体描述 1 游戏过程 取消原来的跳跃和俯身按钮,保留跳跃的功能,可以触屏滑动来躲避地面障碍物,也可以躲避另一种陷阱障碍物 2 闯关功能 取消 ...
- MySQL课堂练习 20162315
练习内容 1.参考教材相关代码,提交能连接到world的截图(有学号水印) 2.查询world数据库,获得人口超过500万的所有城市的列表. 3.查询world数据库,获得New Jersey州所有城 ...
- java_web连接SQL_server详细步骤
(1).我用的是Myeclipse,可以直接将sqljdbc4.jar拷到项目文件 (2).点开SQL Server配置管理器 选中SQL Server2008网络配置下的SQLEXPRESS的协议, ...
- 第二阶段Sprint6
昨天:设置统一保存路径为内存卡,实现可以选择播放已有的视频 今天:将“录制”及“保存”整合到一起,修复出现的Bug,使之能够正常运行. 遇到的问题:感觉调的摄像头录制的画面不好,这怎么办啊?
- 【Coursera】支持向量机
一.最大间隔分类器 1. 函数间隔:\(γ^{i} = y^{i}(w^{T} x + b)\), 改变w和b的量级,对分类结果不会产生任何影响,但是会改变函数间隔的大小.因此,直接对函数间隔求最大值 ...
- Unity控件ScrollView使用问题记录
Unity版本:5.6.2 控件Scroll View由4部分组成,如图: 1.含有Scroll Rect组件的根节点:Scroll View 2.含有Mask组件的节点:Viewport 3.所有内 ...
- [转]让opencv输出人脸检测的得分(置信率)
转自:http://www.cnblogs.com/sciencefans/ 作者:sciencefans 最近项目略多,其中一个需要找出一些和脸比较像但是不是脸的负样本,想用opencv的人脸检测器 ...