Stamp Rally
最大值最小,可以二分,然后并查集看能不能到z个点
但是询问过多,并且发现每次二分要加入的点并不是所有的m条边
于是就考虑整体二分
并查集的处理是重点:
对于一般的dfs分治树,
我们必然要在处理前面部分回溯来的时候,递归右子树之前,左子树并查集的信息必须保留。
但是还要删除当前层的部分并查集的合并操作。
如果直接路径压缩+暴力重构的话,到了后面,每次重构就是O(n)的了,直接N^2了。
为了支持删除,就考虑按秩合并。
合并成功的时候,用一个栈记录pair,然后删除的时候弹栈删除即可。相当于时光倒流
当然,之前的合并一定不能撤销的。
出错点:
1.l打成1海星。。。
2.如果当前区间没有决策位置的话,可以直接返回,但是这些[l,r]编号的边不能扔掉,必须直接加进去。
#include<bits/stdc++.h>
#define il inline
#define reg register int
#define numb (ch^'0')
using namespace std;
typedef long long ll;
il void rd(int &x){
char ch;bool fl=false;
while(!isdigit(ch=getchar()))(ch=='-')&&(fl=true);
for(x=numb;isdigit(ch=getchar());x=x*+numb);
(fl==true)&&(x=-x);
}
namespace Miracle{
const int N=1e5+;
const int M=1e5+;
struct node{
int x,y;
}e[M];
struct que{
int x1,x2,z;
int id,ans;
}q[M],b[M];
int fa[N];
int sz[N];
int op[N];
int n,m;
int fin(int x){
return fa[x]==x?x:fin(fa[x]);
}
pair<int,int>sta[N];
int top;
void merge(int x,int y,bool fl){
int k1=fin(x),k2=fin(y);
if(k1!=k2){
if(sz[k1]>sz[k2]) swap(k1,k2);
fa[k1]=k2;
sz[k2]+=sz[k1];
if(fl)sta[++top]=make_pair(k1,k2);
}
}
void dele(int x,int y){
fa[x]=x;
sz[y]-=sz[x];
}
void divi(int l,int r,int le,int ri){
//cout<<" l r "<<l<<" "<<r<<" : "<<le<<" and "<<ri<<endl;
if(l>r) return;
if(le>ri){
for(reg i=l;i<=r;++i){
merge(e[i].x,e[i].y,);
}
return;
}
if(l==r){
for(reg i=le;i<=ri;++i){
q[i].ans=l;
}
merge(e[l].x,e[l].y,);
return;
}
int mid=(l+r)>>;
for(reg i=l;i<=mid;++i){
merge(e[i].x,e[i].y,);
}
//cout<<" mid "<<mid<<endl;
int pre=le-,bac=ri+;
for(reg i=le;i<=ri;++i){
int k1=fin(q[i].x1),k2=fin(q[i].x2);
int tmp=;
if(k1!=k2){
tmp=sz[k1]+sz[k2];
}else{
tmp=sz[k1];
}
if(tmp>=q[i].z) b[++pre]=q[i];
else b[--bac]=q[i];
}
for(reg i=le;i<=ri;++i){
q[i]=b[i];
}
//cout<<" pre "<<pre<<" bac "<<bac<<" top "<<top<<endl;
//if(top){
while(top){
dele(sta[top].first,sta[top].second);
--top;
}
//}
divi(l,mid,le,pre);
divi(mid+,r,bac,ri);
}
int main(){
rd(n);rd(m);
for(reg i=;i<=m;++i){
rd(e[i].x);rd(e[i].y);
}
int que;
rd(que);
for(reg i=;i<=que;++i){
rd(q[i].x1);rd(q[i].x2);rd(q[i].z);
q[i].id=i;
}
for(reg i=;i<=n;++i){
fa[i]=i;sz[i]=;//;dep[i]=1;
}
divi(,m,,que);
for(reg i=;i<=que;++i){
op[q[i].id]=q[i].ans;
}
for(reg i=;i<=que;++i){
printf("%d\n",op[i]);
}
return ;
} }
signed main(){
// freopen("data.in","r",stdin);
// freopen("my.out","w",stdout);
Miracle::main();
return ;
} /*
Author: *Miracle*
Date: 2018/12/18 8:37:44
*/
还有一种:
你不是暴力重构会TLE吗?但是我一共就需要循环logn次O(m)地把并查集加入
考虑逐层处理
因为对于整体二分,bfs和dfs的顺序都没有问题。
所以bfs处理整体二分,这样,加入就直接加入了,只要暴力重构O(logn)次。
bfs+路径压缩,理论上可以更快一些。
Stamp Rally的更多相关文章
- AtCoder Grand Contest 002 D - Stamp Rally
Description We have an undirected graph with N vertices and M edges. The vertices are numbered 1 thr ...
- [AGC002D] Stamp Rally
确实有想到重构树,不过没有继续下去的思路. 可能是对重构树的性质不太懂. 这种题目我们可以二分答案,考虑怎么\(check\)呢,整体二分+并查集,建出重构树,找去第一个小于这个数的方点,查询他的子树 ...
- 【做题】agc002D - Stamp Rally——整体二分的技巧
题意:给出一个无向连通图,有\(n\)个顶点,\(m\)条边.有\(q\)次询问,每次给出\(x,y,z\),最小化从\(x\)和\(y\)开始,总计访问\(z\)个顶点(一个顶点只计算一次),经过的 ...
- [AGC002D] Stamp Rally 整体二分+并查集
Description 给你一个n个点m个条边构成的简单无向连通图,有Q组询问,每次询问从两个点x,y走出两条路径,使这两条路径覆盖z个点,求得一种方案使得路径上经过的变的最大编号最小. Input ...
- [AGC002D] Stamp Rally (并查集+整体二分)
Description 给你一个n个点m个条边构成的简单无向连通图,有Q组询问,每次询问从两个点x,y走出两条路径,使这两条路径覆盖z个点,求得一种方案使得路径上经过的变的最大编号最小. Input ...
- 【agc002d】Stamp Rally
题目大意 无向图中,每次询问从x和y分别出发,一共经过z个点,使需要走过编号最大的边最小. 解题思路 对于暴力,我们对于每个询问二分答案ans,将1~ans的边加入,用并查集维护,如果x和y在同一个并 ...
- AtCoder Grand Contest 002
AtCoder Grand Contest 002 A - Range Product 翻译 告诉你\(a,b\),求\(\prod_{i=a}^b i\)是正数还是负数还是零. 题解 什么鬼玩意. ...
- RE:从零开始的AGC被虐(到)生活(不能自理)
RE:从零开始的AGC被虐(到)生活(不能自理) 「一直注视着你,似近似远,总是触碰不到.」 --来自风平浪静的明天 AtCoder Grand Contest 001 B: Mysterious L ...
- AtCoder Grand Contest
一句话题解 QwQ主要是因为这篇文章写的有点长……有时候要找某一个题可能不是很好找,所以写了这个东西. 具体的题意.题解和代码可以再往下翻._(:з」∠)_ AGC 001 C:枚举中点/中边. D: ...
随机推荐
- [持久更新] 剑指offer题目Python做题记录
第一题 题目:在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数. 思路:先快速定位到 ...
- python所遇到的坑
我是在ubuntu中,自带的有python2,python3有安装了anaconda套件,所以python的版本很多,曾经想删除过不用的python. 先执行 sudo apt remove pyth ...
- Python20-Day06
常用模块 一.random模块 import random # print(random.random()) #打印0-1之间的小数 # print(random.randint(1,3)) #大于等 ...
- Objective-C中,类方法的getter和setter可以用点运算符吗?
Objective-C中,对象实例property的getter和setter可以使用点运算符来操作,那么类方法的getter和setter可以使用点运算吗? 答案是肯定的. 看如下代码: #impo ...
- 王者荣耀交流协会final发布第五次scrum例会
1.例会照片 成员高远博,冉华,王磊,王玉玲,任思佳,袁玥,王磊,王超. master:王磊 2.时间跨度 2017年12月5日 18:00 — 18:21,总计21分钟 3.地点 一食堂二楼沙发座椅 ...
- 20135234mqy
北京电子科技学院(BESTI) 实 验 报 告 课程:Java实验 班级:1352 姓名: mqy 学号:20135234 成绩: 指导教师 ...
- python service 服务没有及时响应启动或控制请求
1053错误 代码运行没有问题后,安装服务,然而start 的时候出现错误 1053:服务没有及时响应启动或控制请求(Error 1053: The service did not respond t ...
- 冲刺One之站立会议6 /2015-5-19
2015-5-19 今天把服务器端的界面完善了一下,然后大家查了好多资料,实现了登陆界面实际连接的功能,开始加了一个它和服务器的的跳转,但是分析过后发现这是个没有必要的跳转.登录应该直接转到聊天室的主 ...
- C#获取周一、周日的日期 函数类
#region 得到一周的周一和周日的日期 /// <summary> /// 计算本周的周一日期 /// </summary> ...
- The last time the sprint(最后一个冲刺)
经过一两个月的努力,我们终于是做出来了一点东西,从一开始接触这个项目开始,从完全不知道怎么去入手到跌跌碰碰,再到现在可以拿出来一点东西给别人看,我觉得很开心,或许我的这个成品在别人眼中并不算是什么,但 ...