[2016北京集训测试赛15]statement-[线段树+拆环]
Description

Solution
由于题目要求,将a[i]->b[i](边权为i)后所得的图应该是由森林和环套树组合而成。
假如是树形结构,所有的t[i]就直接在线段树t[i]点的dfs序(即in[t[i]],out[t[i]]区间)处记录t[i]点的深度。
这样,针对所有的f[i],在线段树上查找所有包含in[f[i]]点的区间所记录的最大深度d。(这个深度就是在离f[i]最近并且已经验证了是真命题的祖先的深度)
然后用倍增算出f[i]向上到深度d,所经过的编号最大值c。ans=min(ans,c)。
原因:ans是指,图中存在a[i]->b[i](1<=i<=ans)时该询问刚好出现矛盾。如果ans-=1,则所有的f[i]都无法到达离它最近的并且已经验证了是真命题的祖先点。
环套树结构:我们把环拆成树

其中Ca->Cb的边权还是a->b边的边权。以此类推其他都是这样。(原本环套树上以a,b,c,d,e为根的树还是以a,b,c,d,e为根)如此,假如说t[i]在环上,我们除了在in[t[i]],out[t[i]]区间记录t[i]点的深度,还要在in[C(t[i])],out[C(t[i])]区间记录C(t[i])点的深度。这样,不论是原本以t[i]为根的子树,还是环上的以其他点为根的树的信息都可以更新完毕。
Code
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
using namespace std;
const int N=2e5+;
int n,_n,m,Q,a,b;
int fa1[N<<],val[N<<],cir[N];
int tag[N];//0-not_visit 1-tag the stack 2-out the stack
int rt[N],cnt; int y[N<<],nxt[N<<],h[N<<],tot;
void link(int _x,int _y){y[++tot]=_y;nxt[tot]=h[_x];h[_x]=tot;}
void build(int u)
{
if (!fa1[u]){rt[++cnt]=u;tag[u]=;return;}
if (tag[fa1[u]]==)
{
int x=u;
for(int t=fa1[x];t!=u;t=fa1[t])
{
cir[t]=++n;val[n]=val[t];link(n,x);
fa1[x]=n;x=fa1[x];
}
cir[u]=++n;link(n,x);rt[++cnt]=n;
tag[u]=;
return;
}
tag[u]=;
if (!tag[fa1[u]]) build(fa1[u]);link(fa1[u],u);
tag[u]=;
}
int dfn,in[N<<],out[N<<],dep[N<<],fa[N<<][],mxe[N<<][];
void dfs(int x)
{
in[x]=++dfn;mxe[x][]=val[x];dep[x]=dep[fa[x][]]+;
for (int i=;i<=;i++)
{
fa[x][i]=fa[fa[x][i-]][i-];
mxe[x][i]=max(mxe[x][i-],mxe[fa[x][i-]][i-]);
}
for (int i=h[x];i;i=nxt[i]) fa[y[i]][]=x,dfs(y[i]);
out[x]=dfn;
}
int cur[N<<],mxd[N<<];
void modify(int k,int l,int r,int askx,int asky,int d)
{
if (askx<=l&&r<=asky){if (Q<cur[k]||mxd[k]<d) cur[k]=Q,mxd[k]=d;return;}
int mid=(l+r)/;
if (askx<=mid) modify(k<<,l,mid,askx,asky,d);
if (asky>mid) modify(k<<|,mid+,r,askx,asky,d);
}
int query(int k,int l,int r,int x)
{
int mid=(l+r)/,re=cur[k]==Q?mxd[k]:;
if (l==r) return re;
if (x<=mid) re=max(re,query(k<<,l,mid,x));else re=max(re,query(k<<|,mid+,r,x));
return re;
}
int t,f,ret,ref;
int main()
{
scanf("%d%d",&n,&m);_n=n;
for (int i=;i<=m;i++) {scanf("%d%d",&a,&b);fa1[b]=a;val[b]=i;}
for (int i=;i<=_n;i++) if (!tag[i]) build(i);
for (int i=;i<=cnt;i++) dfs(rt[i]);
scanf("%d",&Q);
while (Q--)
{
scanf("%d",&ret);
for (int i=;i<=ret;i++)
{
scanf("%d",&t),modify(,,n,in[t],out[t],dep[t]);
if (cir[t]) modify(,,n,in[cir[t]],out[cir[t]],dep[cir[t]]);
}
int ans=m+,d,c;
scanf("%d",&ref);
for (int i=;i<=ref;i++)
{
scanf("%d",&f);
d=query(,,n,in[f]);c=;
if (d==) continue;else d=dep[f]-d;
for(int j=;d;d>>=,j++)
if (d&) c=max(c,mxe[f][j]),f=fa[f][j];
ans=min(ans,c);
}
if (ans==m+) printf("OK\n");else printf("%d\n",ans);
}
}
[2016北京集训测试赛15]statement-[线段树+拆环]的更多相关文章
- [2016北京集训测试赛5]小Q与内存-[线段树的神秘操作]
Description Solution 哇真的异常服气..线段树都可以搞合并和拆分的啊orzorz.神的世界我不懂 Code #include<iostream> #include< ...
- 2016北京集训测试赛(十七)Problem C: 数组
Solution 线段树好题. 我们考虑用last[i]表示\(i\)这个位置的颜色的上一个出现位置. 考虑以一个位置\(R\)为右端点的区间最远能向左延伸到什么位置: \(L = \max_{i \ ...
- 2016北京集训测试赛(十六)Problem A: 任务安排
Solution 这道题告诉我们, 不能看着数据范围来推测正解的时间复杂度. 事实证明, 只要常数足够小, \(5 \times 10^6\)也是可以跑\(O(n \log n)\)算法的!!! 这道 ...
- 【2016北京集训测试赛(十)】 Azelso (期望DP)
Time Limit: 1000 ms Memory Limit: 256 MB Description 题解 状态表示: 这题的状态表示有点难想...... 设$f_i$表示第$i$个事件经过之 ...
- 【2016北京集训测试赛(二)】 thr (树形DP)
Description 题解 (这可是一道很早就碰到的练习题然后我不会做不想做,没想到在Contest碰到欲哭无泪......) 题目大意是寻找三点对的个数,使得其中的三个点两两距离都为d. 问题在于 ...
- 【2016北京集训测试赛(八)】 crash的数列 (思考题)
Description 题解 题目说这是一个具有神奇特性的数列!这句话是非常有用的因为我们发现,如果套着这个数列的定义再从原数列引出一个新数列,它居然还是一样的...... 于是我们就想到了能不能用多 ...
- 【2016北京集训测试赛(十六)】 River (最大流)
Description Special Judge Hint 注意是全程不能经过两个相同的景点,并且一天的开始和结束不能用同样的交通方式. 题解 题目大意:给定两组点,每组有$n$个点,有若干条跨组 ...
- 【2016北京集训测试赛】river
HINT 注意是全程不能经过两个相同的景点,并且一天的开始和结束不能用同样的交通方式. [吐槽] 嗯..看到这题的想法的话..先想到了每个点的度为2,然后就有点不知所措了 隐隐约约想到了网络流,但并没 ...
- 【2016北京集训测试赛】azelso
[吐槽] 首先当然是要orzyww啦 以及orzyxq奇妙顺推很强qwq 嗯..怎么说呢虽然说之前零零散散做了一些概d的题目但是总感觉好像并没有弄得比较明白啊..(我的妈果然蒟蒻) 这题的话可以说是难 ...
随机推荐
- [WPF 容易忽视的细节] —— x:Name与Name属性
一.前言 WPF使用XAML来对界面进行编写,界面与后台逻辑分离.我们也可以写Style.Trigger来实现一些界面效果, 这些都是通过Name来定位控件的,例如Setter.TargetName. ...
- java io详解(1)
一.java io结构图 二.java io的开始:文件 三.字节流: 一.java io结构图 流分类: 1.Java的字节流 InputStream是所有字节输入流的祖先,而OutputSt ...
- [SQLServer大对象]——FileTable初体验 (转载)
阅读导航启用FILESTREAM设置更改FILESTRAM设置启用数据库非事务性访问级别FileTable 在我接触FileTable之前,存储文件都是存储文件的链接和扩展名到数据,其实并没有实际的把 ...
- 减少MySQL的Sleep进程有效方法
经常遇到很多朋友问到,他的MySQL中有很多Sleep进程,严重占用MySQL的资源,现在分析一下出现这种现象的原因和解决办法: 1,通常来说,MySQL出现大量Sleep进程是因为采用的PHP的My ...
- 全球首款完全开源的堡垒机,符合 4A 的专业运维审计系统Jumpserver
Jumpserver是全球首款完全开源的堡垒机,是符合 4A 的专业运维审计系统. http://www.jumpserver.org https://github.com/jumpserver/ju ...
- python web编程CGI
CGI(通用网关接口),CGI 是Web 服务器运行时外部程序的规范,按CGI 编写的程序可以扩展服务器功能. CGI 应用程序能与浏览器进行交互,还可通过数据库API 与数据库服务器等外部数据源进行 ...
- Matlab feval函数(转)
http://zhidao.baidu.com/link?url=7CusQYQXhCDB8sUtolMEhI1ctnpblbYrpSnU0fhIh5LvDZuhsBuozQusS6Kb1McTp7x ...
- tplink-ssh登录
同步发表:https://www.eatm.app/archives/395.html 备份配置信息 开启SSH #修改文件userconfig/etc/config/dropbear, #查看opt ...
- 【8】python文件的读写方法
(1).读文件的步骤: (1)打开文件 open(path,flag,encoding,[errors]) path:打开路径 flag:打开方式 r(只读) rb(二进制格式) r+(可以读写) w ...
- 团队作业——Alpha冲刺 5/12
团队作业--Alpha冲刺 冲刺任务安排 杨光海天 今日任务:编辑界面完成部分内容,学习了下拉菜单控件的建立,完善界面标题内容,以及交互. 明日任务:继续完善编辑界面,学习使用gallery,着手配图 ...