[Ahoi2005]LANE 航线规划
题目描述
在5个星球之间,有5条探险航线。 A、B两星球之间,如果某条航线不存在,就无法从A星球抵达B星球,我们则称这条航线为关键航线。 显然上图中,1号与5号星球之间的关键航线有1条:即为4-5航线。 然而,在宇宙中一些未知的磁暴和行星的冲撞,使得已有的某些航线被破坏,随着越来越多的航线被破坏,探险飞船又不能及时回复这些航线,可见两个星球之间的关键航线会越来越多。 假设在上图中,航线4-2(从4号星球到2号星球)被破坏。此时,1号与5号星球之间的关键航线就有3条:1-3,3-4,4-5。 小联的任务是,不断关注航线被破坏的情况,并随时给出两个星球之间的关键航线数目。现在请你帮助完成。输入
输出
样例输入
样例输出
首先,题意是求动态图的两点桥数。
将边删除会很麻烦,所以先将要删去的边去掉,形成最后的图。
求边双联通分量,缩点,形成一棵树(一定是一棵树)树边为关键边。
再根据数据加边,每加入一边,则形成一个环,则环上路不再为关键边,最后将答案逆序输出。
加入边后,两点之间的路径中的关键边可用树剖和线段树维护。
线段树中,1表是关键边,0表不是,维护和。每加入一边,求两点LCA,并将线段树相应区间归零(要做延迟表记)。查询就输出线段树相应区间的和
注意细节,代码有点长
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
#include<map>
#include<set>
using namespace std;
struct Messi1
{
int next,to;
}edge1[],edge2[];
struct Messi2
{
int p,q,c;
}a[],b[];
int isbridge[],son[],size[],fa[],dep[],top[],pos[],tot,m;
int head1[],head2[],num1,num2,dfn[],low[],dfscnt,sccno[],n,scccnt,ans[];
int lazy[],c[],cx;
typedef pair<int,int> dou;//定义数对
set<dou> vis,vis2;//此处用set可节约内存,且方便。
void add1(int u,int v)//原图
{
num1++;
edge1[num1].next=head1[u];
edge1[num1].to=v;
head1[u]=num1;
}
void add2(int u,int v)//缩点后的图
{
num2++;
edge2[num2].next=head2[u];
edge2[num2].to=v;
head2[u]=num2;
}
void tarjan(int x,int fa)
{int i;
dfn[x]=low[x]=++dfscnt;
int child=;
for (i=head1[x]; i; i=edge1[i].next)
{
int v=edge1[i].to;
if (dfn[v]==)
{
child++;
tarjan(v,x);
low[x]=min(low[x],low[v]);
if (dfn[x]<low[v]) isbridge[i]=;
}
else if (v!=fa) low[x]=min(low[x],dfn[v]);
}
}
void dfs(int x,int v)
{int i;
//cout<<x<<' '<<v<<endl;
dfn[x]=;
sccno[x]=v;
for (i=head1[x]; i; i=edge1[i].next)
if (isbridge[i]==&&dfn[edge1[i].to]==)
{
dfs(edge1[i].to,v);
}
}
void make_tree()
{int i,j,x1,y1;
tarjan(,-);
memset(dfn,,sizeof(dfn));
for (i=; i<=n; i++)
if (dfn[i]==)
{
scccnt++;
dfs(i,scccnt);
}
for (i=; i<=n; i++)
{
for (j=head1[i]; j; j=edge1[j].next)
{
int v=edge1[j].to;
if (sccno[i]!=sccno[v])
{if (sccno[i]>sccno[v]) x1=sccno[v],y1=sccno[i];
else y1=sccno[v],x1=sccno[i];
if (vis2.find(make_pair(x1,y1))==vis2.end())
//注:set的find函数返回指针,不存在时反回最后节点子节点,
所以不能写!find()。用end函数反回最后节点子节点
{
add2(x1,y1);
add2(y1,x1);
vis2.insert(make_pair(x1,y1));
}
}
}
}
}
void dfs1(int u,int pa,int depth)
{
son[u]=;
size[u]=;
fa[u]=pa;
dep[u]=depth;
for (int j=head2[u]; j; j=edge2[j].next)
{
int v=edge2[j].to;
if (v!=pa)
{
dfs1(v,u,depth+);
size[u]+=size[v];
if (size[v]>size[son[u]]) son[u]=v;
}
}
}
void dfs2(int u,int tp)
{
top[u]=tp;
pos[u]=++tot;
if (son[u]) dfs2(son[u],tp);
for (int j=head2[u]; j; j=edge2[j].next)
if (edge2[j].to!=son[u]&&edge2[j].to!=fa[u])
{
dfs2(edge2[j].to,edge2[j].to);
}
}
void pushdown(int rt)
{
if (lazy[rt])
{
c[rt*]=;
lazy[rt*]=;
c[rt*+]=;
lazy[rt*+]=;
lazy[rt]=;
}
}
int ask(int rt,int l,int r,int L,int R)
{
if (l!=r) pushdown(rt);
if (l>=L&&r<=R)
{
return c[rt];
}
pushdown(rt);
int mid=(l+r)/,s=;
if (L<=mid) s+=ask(rt*,l,mid,L,R);
if (R>mid) s+=ask(rt*+,mid+,r,L,R);
return s;
}
int query(int x,int y)
{
int s=;
while (top[x]!=top[y])
{
if (dep[top[x]]<dep[top[y]]) swap(x,y);
s+=ask(,,n,pos[top[x]],pos[x]);
x=fa[top[x]];
}
if (dep[x]>dep[y]) swap(x,y);
if (x!=y) s+=ask(,,n,pos[x]+,pos[y]);
return s;
}
void update(int rt,int l,int r,int L,int R)
{
if (l!=r)pushdown(rt);
if (l>=L&&r<=R)
{
c[rt]=;
lazy[rt]=;
return;
}
pushdown(rt);
int mid=(l+r)/;
if (L<=mid) update(rt*,l,mid,L,R);
if (R>mid) update(rt*+,mid+,r,L,R);
c[rt]=c[rt*]+c[rt*+];
}
void change(int x,int y)
{
while (top[x]!=top[y])
{
if (dep[top[x]]<dep[top[y]]) swap(x,y);
update(,,n,pos[top[x]],pos[x]);
x=fa[top[x]];
}
if (dep[x]>dep[y]) swap(x,y);
if (x!=y) update(,,n,pos[x]+,pos[y]);
}
void build(int rt,int l,int r)
{
if (l==r)
{
c[rt]=;
return;
}
int mid=(l+r)/;
build(rt*,l,mid);
build(rt*+,mid+,r);
c[rt]=c[rt*]+c[rt*+];
}
int main()
{int i,j,x,y;
//freopen("lane.in","r",stdin);
//freopen("lane.out","w",stdout);
cin>>n>>m;
for (i=; i<=m; i++)
{
scanf("%d%d",&a[i].p,&a[i].q);
vis.insert(make_pair(a[i].p,a[i].q));
}
int tot1=;
scanf("%d",&cx);
while (cx!=-)
{
scanf("%d%d",&x,&y);
if (cx==)
{
tot1++;
b[tot1].p=x;
b[tot1].q=y;
b[tot1].c=;
vis.erase(make_pair(x,y));
}
else
{
tot1++;
b[tot1].c=;
b[tot1].p=x;
b[tot1].q=y;
}
scanf("%d",&cx);
}
for (i=; i<=m; i++)
{
if (vis.find(make_pair(a[i].p,a[i].q))!=vis.end())
{
add1(a[i].p,a[i].q);
add1(a[i].q,a[i].p);
}
}
make_tree();
dfs1(,,);
dfs2(,);
n=scccnt;
//cout<<scccnt<<endl;
build(,,n);
memset(ans,-,sizeof(ans));
for (i=tot1; i>=; i--)
{
if (b[i].c==)
{
int u=sccno[b[i].p],v=sccno[b[i].q];
change(u,v);
}
else
{
int u=sccno[b[i].p],v=sccno[b[i].q];
ans[i]=query(u,v);
}
}
for (i=; i<=tot1; i++)
if (ans[i]!=-) printf("%d\n",ans[i]);
}
[Ahoi2005]LANE 航线规划的更多相关文章
- BZOJ 1969: [Ahoi2005]LANE 航线规划( 树链剖分 )
首先我们要时光倒流, 倒着做, 变成加边操作维护关键边. 先随意搞出一颗树, 树上每条边都是关键边(因为是树, 去掉就不连通了)....然后加边(u, v)时, 路径(u, v)上的所有边都变成非关键 ...
- 【BZOJ 1969】 1969: [Ahoi2005]LANE 航线规划 (树链剖分+线段树)
1969: [Ahoi2005]LANE 航线规划 Description 对Samuel星球的探险已经取得了非常巨大的成就,于是科学家们将目光投向了Samuel星球所在的星系——一个巨大的由千百万星 ...
- 【BZOJ1969】[Ahoi2005]LANE 航线规划 离线+树链剖分+线段树
[BZOJ1969][Ahoi2005]LANE 航线规划 Description 对Samuel星球的探险已经取得了非常巨大的成就,于是科学家们将目光投向了Samuel星球所在的星系——一个巨大的由 ...
- BZOJ1969: [Ahoi2005]LANE 航线规划(LCT)
Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 587 Solved: 259[Submit][Status][Discuss] Description ...
- 【刷题】BZOJ 1969 [Ahoi2005]LANE 航线规划
Description 对Samuel星球的探险已经取得了非常巨大的成就,于是科学家们将目光投向了Samuel星球所在的星系--一个巨大的由千百万星球构成的Samuel星系. 星际空间站的Samuel ...
- 【bzoj1959】[Ahoi2005]LANE 航线规划 树链剖分+线段树
题目描述 对Samuel星球的探险已经取得了非常巨大的成就,于是科学家们将目光投向了Samuel星球所在的星系——一个巨大的由千百万星球构成的Samuel星系. 星际空间站的Samuel II巨型计算 ...
- [bzoj1969] [Ahoi2005]LANE 航线规划
tarjan.并查集.树状数组.树链剖分. 时间倒流,变删边为加边. 先求一波边双联通分量,缩点. 题目保证最后还是整张图联通的..所以就是一棵树. 现在的操作就是,将路径上的边权置0(加边时),查询 ...
- BZOJ 1969: [Ahoi2005]LANE 航线规划 [树链剖分 时间倒流]
题意: 一张图,删除边,求两点之间的割边数量.保证任意时刻图连通 任求一棵生成树,只有树边可能是割边 时间倒流,加入一条边,就是两点路径上的边都不可能是割边,区间覆盖... 然后本题需要把边哈希一下, ...
- ●BZOJ 1969 [Ahoi2005]LANE 航线规划
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=1969 题解: 线段树,树链剖分,反向考虑思路是很巧妙,但是感觉代码真的恶心.. 反着考虑,先 ...
随机推荐
- B-end
Beta冲刺成员名单和工作量比例 姓名 学号 负责内容 工作量比例 张梨贤 170327109 负责企业人员的委托/收回授权.第三方机构的委托授权管理.分级统计展示.分级列表展示 26% 黄腾飞 17 ...
- 简单的C语言编译器--语义制导翻译
语法分析是最难写的,而这部分确实最伤脑的.大量的语义动作分析差点把我逼疯. 简而言之,这部分的作用就是在每次归约之后,都进行一些语义动作,最终让我们得到测试程序的三地址码,即中间代码. 1. ...
- bzoj千题计划275:bzoj4817: [Sdoi2017]树点涂色
http://www.lydsy.com/JudgeOnline/problem.php?id=4817 lct+线段树+dfs序 操作1:access 操作2:u到根的-v到根的-lca到根的*2+ ...
- Scala Option类型
转载自: Scala 初学者指南, 这里有一系列很棒的文章 类型 Option 可能你已经见过它在 Map API 中的使用:在实现自己的提取器时,我们也用过它, 然而,它还需要更多的解释. 你可能会 ...
- 使用Google 的 gson方式解析json
gson支持解析的类型还是比较全面的,包括JavaBean,List<JavaBean>,List<String>,Map等,使用起来也是比较方便,下面根据代码示例给出总结: ...
- CentOS搭建Git服务器及权限管理
声明:本教程,仅作为配置的记录,细节不展开,需要您有一点linux的命令基础,仅作为配置参考. 1. 系统环境 系统: Linux:CentOS 7.2 64位 由于CentOS已经内置了OpenSS ...
- Python基础学习篇章三
一. Python对象类型 1. 对象是Python最基本的概念,一个Python程序可以分解为模块.语句.表达式.和对象.它们的关系如下:(1)程序由模块构成 (2)模块包含语句 (3)语句包含表达 ...
- ELK学习总结(2-3)Mget获取多个文档
mget 获取多个文档 1.curl 命令格式:mget获取多个文档: curl 'localhost:9200/_mget' -d '{ "docs":[ { " ...
- python flask框架 tempates 模版的使用
在py文件同级下 建立templates文件夹,再文件夹中编写html文件 1 向模版中传递参数: ''' 1 向模板传送 参数 ''' @app.route('/') def index(): na ...
- Python之编码
一.Python2与Python3的区别 1.从宏观上考虑,Python2重复代码太多,错误率高,不够规范.Python崇尚的是语言简洁.优美.清晰.Python3更加规范,重复代码少: 2.Pyth ...