BZOJ2733 [HNOI2012]永无乡(并查集+线段树合并)
题目大意:
在$n$个带权点上维护两个操作:
1)在点$u,v$间连一条边;
2)询问点$u$所在联通块中权值第$k$小的点的编号,若该联通块中的点的数目小于$k$,则输出$-1$;
上周的模拟赛在一道线段树合并的题目上gg了,来学习一个。
对每一个联通块,我们维护一棵权值线段树。查询时,若左子树大小大于等于$k$进入左子树,否则进入右子树;
因为每棵线段树同构,所以对于任意两棵线段树可以进行合并操作:
int merge(int x,int y){
if(!x)return y;
if(!y)return x;
t[x].lson=merge(t[x].lson,t[y].lson);
t[x].rson=merge(t[x].rson,t[y].rson);
t[x].s=t[t[x].lson].s+t[t[x].rson].s;
return x;
}
利用并查集判断两个点是否连通,若不联通,则合并两个联通块的线段树即可;
代码:
#include<cstring>
#include<cmath>
#include<cstdio>
#include<algorithm>
#include<cctype>
#define foru(i,x,y) for(int i=x;i<=y;i++)
using namespace std;
const int N=1e5+; struct node{int s,lson,rson;}t[N*];
int n,m,a[N],f[N],fx,fy,cnt,rt[N],idx[N]; int gf(int k){return k==f[k]?k:f[k]=gf(f[k]);} int read(){
static int f,x;static char ch;
x=f=;ch=getchar();
while(!isdigit(ch)){f=(ch=='-');ch=getchar();}
while(isdigit(ch)){x=x*+ch-'';ch=getchar();}
return f?-x:x;
} #define ls t[k].lson
#define rs t[k].rson
#define mid ((L+R)>>1) void upd(int &k,int L,int R,int p){
if(p<L||p>R)return;
if(!k)k=++cnt;
if(L==R){t[k].s=;return;}
upd(ls,L,mid,p);upd(rs,mid+,R,p);
t[k].s=t[ls].s+t[rs].s;
} int query(int k,int L,int R,int p){
if(L==R)return L;
if(t[ls].s<p)return query(rs,mid+,R,p-t[ls].s);
return query(ls,L,mid,p);
} int merge(int x,int y){
if(!x)return y;
if(!y)return x;
t[x].lson=merge(t[x].lson,t[y].lson);
t[x].rson=merge(t[x].rson,t[y].rson);
t[x].s=t[t[x].lson].s+t[t[x].rson].s;
return x;
} int main(){
//freopen("1.in","r",stdin);
memset(t,,sizeof(t));
n=read();m=read();
foru(i,,n)a[i]=read(),f[i]=i;
foru(i,,m){
fx=gf(read());fy=gf(read());
f[fx]=fy;
}
foru(i,,n){
upd(rt[gf(i)],,n,a[i]);
idx[a[i]]=i;
}
int q=read(),x,y;
char ch[];
while(q--){
scanf("%s%d%d",ch,&x,&y);
fx=gf(x);fy=gf(y);
if(ch[]=='Q')
printf("%d\n",t[rt[fx]].s>=y?idx[query(rt[fx],,n,y)]:-);
else if(fx!=fy){
f[fx]=fy;
rt[fy]=merge(rt[fx],rt[fy]);
}
}
return ;
}
BZOJ2733 [HNOI2012]永无乡(并查集+线段树合并)的更多相关文章
- B20J_2733_[HNOI2012]永无乡_权值线段树合并
B20J_2733_[HNOI2012]永无乡_权值线段树合并 Description:n座岛,编号从1到n,每座岛都有自己的独一无二的重要度,按照重要度可以将这n座岛排名,名次用1到 n来表示.某些 ...
- bzoj 2733 永无乡 - 并查集 - 线段树
永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接,通过桥可以从一个岛 到达另一个岛. ...
- [bzoj2733][HNOI2012]永无乡_权值线段树_线段树合并
永无乡 bzoj-2733 HNOI-2012 题目大意:题目链接. 注释:略. 想法: 它的查询操作非常友善,就是一个联通块内的$k$小值. 故此我们可以考虑每个联通块建一棵权值线段树. 这样的话每 ...
- 【洛谷P3224】永无乡 并查集+Splay启发式合并
题目大意:给定 N 个点的图,点有点权,初始有一些无向边,现在有 Q 个询问,每个询问支持动态增加一条无向边连接两个不连通的点和查询第 X 个点所在的联通块中权值第 K 大的是哪个点. 题解:学会了平 ...
- BZOJ 3910 并查集+线段树合并
思路: 1. 并查集+线段树合并 记得f[LCA]==LCA的时候 f[LCA]=fa[LCA] 2.LCT(并不会写啊...) //By SiriusRen #include <cstdio& ...
- bzoj2733 / P3224 [HNOI2012]永无乡(并查集+线段树合并)
[HNOI2012]永无乡 每个联通块的点集用动态开点线段树维护 并查集维护图 合并时把线段树也合并就好了. #include<iostream> #include<cstdio&g ...
- bzoj2733: [HNOI2012]永无乡(splay)
2733: [HNOI2012]永无乡 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 3778 Solved: 2020 Description 永 ...
- bzoj2733: [HNOI2012]永无乡 启发式合并
地址:http://www.lydsy.com/JudgeOnline/problem.php?id=2733 题目: 2733: [HNOI2012]永无乡 Time Limit: 10 Sec ...
- [Bzoj2733][Hnoi2012] 永无乡(BST)(Pb_ds tree)
2733: [HNOI2012]永无乡 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 4108 Solved: 2195[Submit][Statu ...
随机推荐
- 爱奇艺用券付费VIP电影+python爬虫程序+可视化界面+下载本地
申明:本博客中的工具及源码仅供个人学习使用,请勿用作商业等其他任何违法用途!否则后果自负 直接步入正题吧! 工具开发环境:windows10,python3.6 工具界面设计:基于python 自带的 ...
- UML-异常处理
1.名词解释 缺陷(Fault):错误引起的行为.如:程序员拼写错了数据库名称 错误(Error):缺陷在运行系统中的表现.如:当使用拼写错误的名称调用数据库时,抛出数据库异常 故障(Failure) ...
- C语言笔记 15_标准库&locale&math&setjmp&signal&stdarg&stddef
<locale.h> 简介 locale.h 头文件定义了特定地域的设置,比如日期格式和货币符号.接下来我们将介绍一些宏,以及一个重要的结构 struct lconv 和两个重要的函数. ...
- fastreport小入门
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms ...
- dp--树形dp P1352 没有上司的舞会
题目描述 某大学有N个职员,编号为1~N.他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司.现在有个周年庆宴会,宴会每邀请来一个职员都会增加一定的快乐指数Ri, ...
- 吴裕雄--天生自然MySQL学习笔记:MySQL 创建数据库
在登陆 MySQL 服务后,使用 create 命令创建数据库,语法如下: CREATE DATABASE 数据库名; 以下命令简单的演示了创建数据库的过程,数据名为 RUNOOB: [root@ho ...
- SQL基础教程(第2版)第3章 聚合与排序:3-2 对表进行分组
第3章 聚合与排序:3-2 对表进行分组 ● 使用GROUP BY子句可以像切蛋糕那样将表分割.通过使用聚合函数和GROUP BY子句,可以根据“商品种类”或者“登记日期”等将表分割后再进行汇总.● ...
- bootstrap 网格
实现原理 网格系统的实现原理非常简单,仅仅是通过定义容器大小,平分12份(也有平分成24份或32份,但12份是最常见的),再调整内外边距,最后结合媒体查询,就制作出了强大的响应式网格系统.Bootst ...
- 对PHP-GC(垃圾回收)的一点理解
一直对php的垃圾回收机制不明不白故自己开贴记录下. 首先,说到垃圾回收,得先知道什么情况下才能产生垃圾. 如果一个变量refcount在增加,说明在被使用,不是垃圾. 如果一个变量的refcount ...
- 201771010123汪慧和《面向对象程序设计Java》第二周学习总结
一.理论知识部分 1.标识符由字母.下划线.美元符号和数字组成, 且第一个符号不能为数字.标识符可用作: 类名.变量名.方法名.数组名.文件名等.第二部分:理论知识学习部分 2.关键字就是Java语言 ...