[hdu2460]network(依次连边并询问图中割边数量) tarjan边双联通分量+lca
题意:
给定一个n个点m条边的无向图,q个操作,每个操作给(x,y)连边并询问此时图中的割边有多少条。(连上的边会一直存在)
n<=1e5,m<=2*10^5,q<=1e3,多组数据。
题解:
用tarjan求边双连通分量并缩点,缩点后组成一棵树,记录此时割边共有sum条。
连接(x,y),设c[i]为缩点后i所在的新点(边双连通分量),则c[x]-->lca-->c[y]形成一个环,环上的所有边都不再是割边,走一遍并标记,如果遇到没标记过的就sum--。
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
using namespace std; const int N=,M=,K=;
int n,m,al,bl,sum,num,sl,cnt;
int afirst[N],bfirst[N],fa[N][K],dfn[N],low[N],is[N],c[N],s[N],dep[N];
struct node{
int x,y,tmp,next;
}a[*M],b[*M]; int minn(int x,int y){return x<y ? x:y;} void ins_a(int x,int y)
{
a[++al].x=x;a[al].y=y;a[al].tmp=;
a[al].next=afirst[x];afirst[x]=al;
} void ins_b(int x,int y)
{
b[++bl].x=x;b[bl].y=y;
b[bl].next=bfirst[x];bfirst[x]=bl;
} void tarjan(int x)
{
dfn[x]=low[x]=++num;
s[++sl]=x;
for(int i=afirst[x];i;i=a[i].next)
{
if(a[i].tmp) continue;
a[i].tmp=;
a[i%== ? i-:i+].tmp=;
int y=a[i].y;
if(!dfn[y])
{
tarjan(y);
low[x]=minn(low[x],low[y]);
if(dfn[x]<low[y])
{
sum++;//printf("%d -- > %d\n",x,y);
cnt++;
int z;
while()
{
z=s[sl--];
c[z]=cnt;
if(z==y) break;
}
}
}
else low[x]=minn(low[x],dfn[y]);
}
} void dfs(int x)
{
for(int i=bfirst[x];i;i=b[i].next)
{
int y=b[i].y;
if(y==fa[x][]) continue;
fa[y][]=x;dep[y]=dep[x]+;
dfs(y);
}
} void prepare_lca()
{
memset(fa,,sizeof(fa));
memset(dep,,sizeof(dep));
memset(is,,sizeof(is));
dep[]=;
dfs();
for(int j=;j<=;j++)
for(int i=;i<=n;i++)
fa[i][j]=fa[fa[i][j-]][j-];
} int get_lca(int x,int y)
{
if(dep[x]<dep[y]) swap(x,y);
for(int i=;i>=;i--)
if(dep[fa[x][i]]>=dep[y]) x=fa[x][i];
if(x==y) return x;
for(int i=;i>=;i--)
{
if(fa[x][i]!=fa[y][i]) x=fa[x][i],y=fa[y][i];
}
return fa[x][];
} void del(int x,int y,int lca)
{
while(x!=lca)
{
if(!is[x]) sum--;
is[x]=;
x=fa[x][];
}
while(y!=lca)
{
if(!is[y]) sum--;
is[y]=;
y=fa[y][];
}
} int main()
{
freopen("a.in","r",stdin);
int q,x,y,T=;
while()
{
scanf("%d%d",&n,&m);
if(!n && !m) return ;
printf("Case %d:\n",++T);
al=;bl=;sum=;
memset(afirst,,sizeof(afirst));
memset(bfirst,,sizeof(bfirst));
num=;sl=;cnt=;
memset(dfn,,sizeof(dfn));
memset(c,,sizeof(c));
for(int i=;i<=m;i++)
{
scanf("%d%d",&x,&y);
ins_a(x,y);ins_a(y,x);
}
for(int i=;i<=n;i++)
{
if(!dfn[i])
{
tarjan(i);
if(sl)
{
cnt++;
for(int i=;i<=sl;i++) c[s[i]]=cnt;
sl=;
}
}
} for(int i=;i<=*m;i++)
{
x=a[i].x,y=a[i].y;
if(c[x]!=c[y]) ins_b(c[x],c[y]);
}
// for(int i=1;i<=bl;i++)
// printf("%d -- > %d\n",b[i].x,b[i].y);
// for(int i=1;i<=n;i++) printf("c [ %d ] = %d\n",i,c[i]);
prepare_lca();
scanf("%d",&q);
for(int i=;i<=q;i++)
{
scanf("%d%d",&x,&y);
x=c[x];y=c[y];
int lca=get_lca(x,y);
del(x,y,lca);
printf("%d\n",sum);
}
printf("\n");
}
return ;
}
[hdu2460]network(依次连边并询问图中割边数量) tarjan边双联通分量+lca的更多相关文章
- [J]computer network tarjan边双联通分量+树的直径
https://odzkskevi.qnssl.com/b660f16d70db1969261cd8b11235ec99?v=1537580031 [2012-2013 ACM Central Reg ...
- POJ3694 Network —— 边双联通分量 + 缩点 + LCA + 并查集
题目链接:https://vjudge.net/problem/POJ-3694 A network administrator manages a large network. The networ ...
- 图连通性【tarjan点双连通分量、边双联通分量】【无向图】
根据 李煜东大牛:图连通性若干拓展问题探讨 ppt学习. 有割点不一定有割边,有割边不一定有割点. 理解low[u]的定义很重要. 1.无向图求割点.点双联通分量: 如果对一条边(x,y),如果low ...
- poj 3177 Redundant Paths 求最少添加几条边成为双联通图: tarjan O(E)
/** problem: http://poj.org/problem?id=3177 tarjan blog: https://blog.csdn.net/reverie_mjp/article/d ...
- UML类图中的关系和表示方法
类图是用来描述程序中的类以及它们之间的关系的,使用类图可以帮助我们简化对系统的理解.在UML类图中比较常见的关系有六种,它们分别是:依赖.关联.聚合.组合.泛化.实现,这六种关系中类之间的紧密程度是依 ...
- Ex 3_25 图中每个顶点有一个相关价格..._十一次作业
(a)首先对有向无环图进行拓扑排序,再按拓扑排序的逆序依次计算每个顶点的cost值,每个顶点的cost值为自身的price值与相邻顶点间的cost值得最小值 (b)求出图中的每一个强连通分量,并把所有 ...
- Dijkstra 算法,用于对有权图进行搜索,找出图中两点的最短距离
Dijkstra 算法,用于对有权图进行搜索,找出图中两点的最短距离,既不是DFS搜索,也不是BFS搜索. 把Dijkstra 算法应用于无权图,或者所有边的权都相等的图,Dijkstra 算法等同于 ...
- UML类图中的各种箭头代表的含义(转自:http://www.cnblogs.com/damsoft/archive/2016/10/24/5993602.html)
1.UML简介Unified Modeling Language (UML)又称统一建模语言或标准建模语言. 简单说就是以图形方式表现模型,根据不同模型进行分类,在UML 2.0中有13种图,以下是他 ...
- UML类图中的关系表示
UML类图中的关系和表示方法 类图是用来描述程序中的类以及它们之间的关系的,使用类图可以帮助我们简化对系统的理解.在UML类图中比较常见的关系有六种,它们分别是:依赖.关联.聚合.组合.泛化.实现,这 ...
随机推荐
- (转)apktool+dex2jar+jd_gui
转:http://www.cnblogs.com/MichaelGuan/archive/2011/10/25/2224578.html apktool: 可以解析资源文件,比如布局文件xml等,方便 ...
- sping框架(3)— 使用spring容器
spring有两个核心接口:BeanFactory和ApplicationContext,其中ApplicationContext是BeanFactory的子接口.它们都可以代表spring容器,sp ...
- iOS- 给App添加内购& 验证购买iOS7新特性
1.内购——应用内购买 我所说的内购——也可以说是应用内购买 大家都知道通过苹果应用程序商店有三种主要赚钱的方式: 1.直接收费(与国内大部分用户的消费习惯相悖,如果要收费,直接收高的,别收6块钱) ...
- Python 时间推进器-->在当前时间的基础上推前n天 | CST时间转化标准日期格式
由于公司任务紧迫,好久没有在园子里写自己的心得了,今天偷个闲发表点简单的代码块,在开源的时代贡献微薄力量.话不多说,直接上代码块: ]) m = ]) d = ]) the_date = dateti ...
- SQL SERVER 存储过程中SELECT 返回值如何赋值给变量
今天在处理一个问题时,使用到一个存储过程,是用于更新并获取最新ID的.在使用过程中,需要获取到这个ID并赋值给变量,结果用EXEC @ID = 存储过程的方式获取失败了.具体情况如下: 为了还原整个情 ...
- 单点登录Windows实现
Windows实现步骤: server.xml修改方式 hosts修改方式 CAS客户端配置 CAS配置filter.txt内容如下: <!-- ======================== ...
- [NOIP2017]逛公园 最短路图 拓扑序DP
---题面--- 题解: 挺好的一道题. 首先我们将所有边反向,跑出n到每个点的最短路,然后f[i][j]表示从i号节点出发,路径长比最短路大j的方案数. 观察到,如果图中出现了0环,那么我们可以通过 ...
- 【POJ3621】【洛谷2868】Sightseeing Cows(分数规划)
[POJ3621][洛谷2868]Sightseeing Cows(分数规划) 题面 Vjudge 洛谷 大意: 在有向图图中选出一个环,使得这个环的点权\(/\)边权最大 题解 分数规划 二分答案之 ...
- 20165218 2017-2018-1 《Java程序设计》第四周学习总结
20165218 2017-2018-1 <Java程序设计>第四周学习总结 教材学习内容总结 第五章 子类与继承 子类与父类 通过关键字extands定义子类 class 子类 exta ...
- POJ.2142 The Balance (拓展欧几里得)
POJ.2142 The Balance (拓展欧几里得) 题意分析 现有2种质量为a克与b克的砝码,求最少 分别用多少个(同时总质量也最小)砝码,使得能称出c克的物品. 设两种砝码分别有x个与y个, ...