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: ...
随机推荐
- LINUX开发使用的3个远程工具
1,SecureCRT 2,SSH Secure Shell Client 3,VNC Viewer 如果想VNC Server启动时加载vncserver服务 需要修改/etc/rc.d/rc.lo ...
- mysql和oracle查询出的一条结果中的多个字段拼接
1,mysql concat('a','b','c')和concat_ws('a','b','c')的区别:前者如果有某个值为空,结果为空;后者如果有某个值为空,可以忽略这个控制 SELECT con ...
- Daily Scrum (2015/10/21)
今天可以说是项目正式开始的第一天,由于大家缺乏做团队项目的经验,对TFS的使用都还不太熟悉,所以今天大家的主要工作是熟悉TFS的使用和对代码进行初步的理解.我们预计需要2-3天时间来理解透彻源代码.以 ...
- 20172308 实验一《Java开发环境的熟悉》实验报告
20172308 2017-2018-2 <程序设计与数据结构>实验1报告 课程:<程序设计与数据结构> 班级: 1723 姓名: 周亚杰 学号:20172308 实验教师:王 ...
- 第二阶段Sprint冲刺会议10
进展:把所有功能整合到主界面,结果导致视频只能播放不能录制,闹钟加不进去,导致闹钟功能差点不能用,放弃整合.
- spring冲刺计划
会议召开时间表 日期 时间 内容 05/09 21:00-22:00 讨论题目(未果) 05/10 21:00-21:30 确定题目(网络助手) 05/13 21:00-21:45 讨论软件页面设计 ...
- c++ imooc自学计划
一.视频学习相关的课程列表: C++远征之起航篇http://www.imooc.com/learn/342: C++远征之离港篇http://www.imooc.com/learn/381: C++ ...
- Scanner的例子
package com.firstDay.one; import java.util.Scanner; public class Information { /** * @param args */ ...
- Beta 冲刺 (3/7)
队名:日不落战队 安琪(队长) 过去两天完成了那些任务 上传个人信息. 接下来的任务 建立和上传收藏夹. 还剩下的任务 完善手写涂鸦. 社交模块. 遇到的困难 暂无. 有哪些收获和疑问 收获:okht ...
- 简单复利计算java板
一.要求: 1.客户说:帮我开发一个复利计算软件. 2如果按照单利计算,本息又是多少呢? 3.假如30年之后要筹措到300万元的养老金,平均的年回报率是3%,那么,现在必须投入的本金是多少呢? 4.利 ...