[APIO2019] [LOJ 3145] 桥梁(分块+并查集)(有详细注释)
[APIO2019] [LOJ 3145] 桥梁(分块+并查集)(有详细注释)
题面
略
分析
考试的时候就感觉子任务4是突破口,结果却写了个Kruskal重构树,然后一直想怎么在线用数据结构维护
实际上是离线算法。考虑只有查询的时候。我们可以离线对查询的权值从大到小排序,边也按边权从大到小排序,然后对于权值比询问大的边,把边两端结点集合合并。答案就是查询点所在点集的大小。只需要用并查集维护,然后双指针扫描,由于一条边只会被加进去一次,时间复杂度为$ O(n\log n)$
考虑有修改的情况。所以我们可以对查询分块(按时间每B个询问分为一块),然后对每个块里的询问按上述方法暴力维护。最后还要更新块内修改操作,因为这些修改会对下一块有影响。注意修改会影响点集联通情况,用可撤销的并查集维护。.(因为每次操作都要把m条边扫描一遍).总复杂度\(O(q\sqrt {q \log n} )\)
细节较多,注释见代码
代码
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#define maxn 500000
#define maxm 1000000
#define bsz 1000
using namespace std;
inline void qread(int &x){
x=0;
int sign=1;
char c=getchar();
while(c<'0'||c>'9'){
if(c=='-') sign=-1;
c=getchar();
}
while(c>='0'&&c<='9'){
x=x*10+c-'0';
c=getchar();
}
x=x*sign;
}
inline void qprint(int x){
if(x<0){
putchar('-');
qprint(-x);
}else if(x==0){
putchar('0');
return;
}else{
if(x>=10) qprint(x/10);
putchar(x%10+'0');
}
}
int n,m,q;
struct edge{
int from;
int to;
int val;
int id;
}E[maxm+5];
int fa[maxn+5];
int sz[maxn+5];
int find(int x){
// printf("%d\n",x);
if(fa[x]==x) return x;
else return find(fa[x]);
}
void ini(int n){
for(int i=1;i<=n;i++){
fa[i]=i;
sz[i]=1;
}
}
int bel[maxn+5];
int lastx[maxm+5],lasty[maxm+5];//用于记录并查集之前状态,回滚时用
void merge(int id){
int x=E[id].from;
int y=E[id].to;
int fx=find(x),fy=find(y);
if(fx!=fy){
if(sz[fx]>sz[fy]) swap(fx,fy);
lastx[id]=fx;
lasty[id]=fy;
fa[fx]=fy;
sz[fy]+=sz[fx];
}
}
struct oper{
int type;
int x;
int val;
}in[maxn+5];
struct query{//询问
int id;
int x;
int val;
query(){
}
query(int _id,int _x,int _val){
id=_id;
x=_x;
val=_val;
}
friend bool operator < (query p,query q){
return p.val>q.val;
}
};
vector<query>Q;
struct update{//需要合并的边
int id;
int val;
update(){
}
update(int _id,int _val){
id=_id;
val=_val;
}
friend bool operator < (update p,update q){
return p.val>q.val;
}
};
int ans[maxn+5];
bool vis[maxn+5];//标记哪些边的边权需要修改
bool tmp[maxn+5];//标记当前块需要修改的边
vector<update>U1;//不需要修改,但是需要合并的边
vector<int>U2;//需要修改,需要合并的边
vector<int>back;//回滚用
void rebuild(int last){
for(int i=1;i<=m;i++){
U1.push_back(update(i,E[i].val));
lastx[i]=0;
}
ini(n);
sort(U1.begin(),U1.end());
sort(Q.begin(),Q.end());
for(int i=0,j=0;i<Q.size();i++){////遍历每个询问
back.clear();
while(j<U1.size()&&U1[j].val>=Q[i].val){//双指针找出可以经过的边
if(!vis[U1[j].id]){
merge(U1[j].id);
lastx[U1[j].id]=0;//不需要回滚
}
j++;
}
for(int p=last;p<Q[i].id;p++){
if(in[p].type==1) tmp[in[p].x]=1;
}
for(int p=0;p<U2.size();p++){//修改前边权比当前询问要大,
//由于在后续可能涉及修改,需要进行回滚
if(!tmp[U2[p]]&&Q[i].val<=E[U2[p]].val){
merge(U2[p]);
back.push_back(U2[p]);
}
}
for(int p=Q[i].id;p>=last;p--){
//查询时序之前被修改了,并且修改后边权比当前查询负载要大
//注意:只有修改后边权比当前查询大的边才会进行合并,所以通过from[s[p]]=-1在回撤时过滤掉不符合要求的边
if(in[p].type==1) tmp[in[p].x]=0;//回滚tmp数组
if(in[p].type==2||lastx[in[p].x]) continue;//跳过query
lastx[in[p].x]=-1;
back.push_back(in[p].x);
if(in[p].val>=Q[i].val) merge(in[p].x);//如果修改后变得对答案有影响,就合并
}
ans[Q[i].id]=sz[find(Q[i].x)];
for(int p=back.size()-1;p>=0;p--){//回滚
if(lastx[back[p]]!=-1){
sz[lasty[back[p]]]-=sz[lastx[back[p]]];
fa[lastx[back[p]]]=lastx[back[p]];
}
lastx[back[p]]=0;
}
}
}
int main(){
qread(n);
qread(m);
for(int i=1;i<=m;i++){
qread(E[i].from);
qread(E[i].to);
qread(E[i].val);
E[i].id=i;
}
qread(q);
for(int i=1;i<=q;i++) bel[i]=i/bsz+1;
int last=1;
for(int i=1;i<=q;i++){
qread(in[i].type);
qread(in[i].x);
qread(in[i].val);
if(in[i].type==1){
if(!vis[in[i].x]) U2.push_back(in[i].x);
vis[in[i].x]=1;
}else{
Q.push_back(query(i,in[i].x,in[i].val));
}
if(bel[i]!=bel[i+1]){
///存满一个块,进行一次离线操作
rebuild(last);
while(last<=i){//last记录修改和查询到哪里
vis[in[last].x]=0;
if(in[last].type==1) E[in[last].x].val=in[last].val;
else{
qprint(ans[last]);
putchar('\n');
}
last++;
}
U1.clear();
U2.clear();
Q.clear();
back.clear();
}
}
}
[APIO2019] [LOJ 3145] 桥梁(分块+并查集)(有详细注释)的更多相关文章
- P5443 [APIO2019]桥梁 [分块+并查集]
分块+并查集,大板子,没了. 并查集不路径压缩,可撤销,然后暴力删除 这样对于每个块都是独立的,所以直接搞就行了. 然后块内修改操作搞掉,就是单独的了 // powered by c++11 // b ...
- 洛谷P3247 最小公倍数 [HNOI2016] 分块+并查集
正解:分块+并查集 解题报告: 传送门! 真的好神仙昂QAQ,,,完全想不出来,,,还是太菜了QAQ 首先还是要说下,这题可以用K-D Tree乱搞过去(数据结构是个好东西昂,,,要多学学QAQ),但 ...
- [BZOJ4537][HNOI2016]最小公倍数(分块+并查集)
4537: [Hnoi2016]最小公倍数 Time Limit: 40 Sec Memory Limit: 512 MBSubmit: 1687 Solved: 607[Submit][Stat ...
- BZOJ4320 ShangHai2006 Homework(分块+并查集)
考虑根号分块.对于<√3e5的模数,每加入一个数就暴力更新最小值:对于>√3e5的模数,由于最多被分成√3e5块,查询时对每一块找最小值,这用一些正常的DS显然可以做到log,但不太跑得过 ...
- HDU 6271 Master of Connected Component(2017 CCPC 杭州 H题,树分块 + 并查集的撤销)
题目链接 2017 CCPC Hangzhou Problem H 思路:对树进行分块.把第一棵树分成$\sqrt{n}$块,第二棵树也分成$\sqrt{n}$块. 分块的时候满足每个块是一个 ...
- bzoj 4537: [Hnoi2016]最小公倍数 分块+并查集
题目大意: 给定一张n个点m条边的无向图,每条边有两种权.每次询问某两个点之间是否存在一条路径上的边的两种权的最大值分别等于给定值. n,q <= 50000. m <= 100000 题 ...
- Codeforces 506D Mr. Kitayuta's Colorful Graph(分块 + 并查集)
题目链接 Mr. Kitayuta's Colorful Graph 把每种颜色分开来考虑. 所有的颜色分为两种:涉及的点的个数 $> \sqrt{n}$ 涉及的点的个数 $<= ...
- 洛谷P4004 Hello world!(分块+并查集)
传送门 虽然洛谷数据水,然而咱最终还是没有卡过uoj上的毒瘤数据-- 神tm全uoj就3个人过了这题-- 首先,每个数最多被开根\(6\)次,开到\(1\)之后就别管它了,把它用并查集连到它父亲上 它 ...
- 并查集 & 最小生成树详细讲解
并查集 & 最小生成树 并查集 Disjoint Sets 什么是并查集? 并查集,在一些有N个元素的集合应用问题中,我们通常是在开始时让每个元素构成一个单元素的集合,然后按一定顺序将 ...
随机推荐
- 唤醒 App
一.Deep Link 1.什么是 Deep Link? Deep Link 是 App 的深度连接,当单击链接或编程请求调用Web URI意图时,Android系统按顺序依次尝试以下每一个操作,直到 ...
- DevExpress ASP.NET Core Controls 2019发展蓝图(No.5)
本文主要为大家介绍DevExpress ASP.NET Core Controls 2019年的官方发展蓝图,更多精彩内容欢迎持续收藏关注哦~ [DevExpress ASP.NET Controls ...
- NOIP模拟赛(by hzwer) T3 小奇回地球
[题目背景] 开学了,小奇在回地球的路上,遇到了一个棘手的问题. [问题描述] 简单来说,它要从标号为 1 的星球到标号为 n 的星球,某一些星球之间有航线. 由于超时空隧道的存在,从一个星球到另一个 ...
- jzoj6404. 【NOIP2019模拟11.04】B
题目描述 Description Input 从文件b.in中读入数据. 第丬行三个正整数 n, m, K. 接下来 n 行每行 m 个正整数, 表示矩阵A. Output 输出到文件b.out中. ...
- vue父组件更新,子组件也更新的方法
1.父组件 使用 Math.ramdom() 2.子组件获取 然后监听这个ramdom变化,处理子组件的更新
- mysql TOP语句 语法
mysql TOP语句 语法 作用:用于规定要返回的记录的数目. 语法:SELECT column_name(s) FROM table_name LIMIT number 说明:对于拥有数千条记录的 ...
- codeforces 819B - Mister B and PR Shifts(思维)
原题链接:http://codeforces.com/problemset/problem/819/B 题意:把一个数列整体往右移k位(大于n位置的数移动到数列前端,循环滚动),定义该数列的“偏差值” ...
- 【CF1247F】Tree Factory(构造)
题意:给定一棵n个点的树,要求将一条可以随意标号的链通过若干次操作变成这棵树 一次操作是指若v不为根且v的父亲不为根,则将v以及v的子树移到v的父亲的父亲上 要求给出标号方案,操作次数以及方案 n&l ...
- Spring Cloud架构教程 (六)消息驱动的微服务【Dalston版】
Spring Cloud Stream是一个用来为微服务应用构建消息驱动能力的框架.它可以基于Spring Boot来创建独立的.可用于生产的Spring应用程序.它通过使用Spring Integr ...
- codeforces D Salary Changing
题意:给你n个人,和s块钱,每个人都有一个工资区间,你给所有人都发工资.然后要他们工资的中位数最大. 思路:二分找那个值.那个值要满足至少有n/2+1个工资区间内. #include<cstdi ...