poj 3694 Network : o(n) tarjan + O(n) lca + O(m) 维护 总复杂度 O(m*q)
/**
problem: http://poj.org/problem?id=3694 问每加一条边后剩下多少桥
因为是无向图,所以使用tarjan缩点后会成一棵树并维护pre数组
在树上连一条边(a,b)减少的桥数就是
a点到a点和b点的最近公共祖先(lca)的所有边+b点到a点和b点的最近公共祖先的所有边
在算桥的同时将这些点缩成一个点
即每个点color = 最近公共祖先color
同时维护pre数组 每个点的pre = 最近公共祖先的pre 即可
**/
#include<stdio.h>
#include<stack>
#include<queue>
#include<algorithm>
using namespace std; const int MAXN = ;
const int MAXM = ; class Graphics{
private:
struct Edge{
int to, next;
bool bridge;
}edge[MAXM];
struct Point{
int dfn, low, color;
}point[MAXN];
int first[MAXN], pre[MAXN], sign, sumOfPoint, dfnNum, colorNum, bridge;
bool vis[MAXN];
stack<int> stk;
queue<int> bfs;
void tarjan(int u, int preEdge = -){
point[u].low = dfnNum;
point[u].dfn = dfnNum ++;
vis[u] = true;
stk.push(u);
for(int i = first[u]; i != -; i = edge[i].next){
int to = edge[i].to;
if((i^) == preEdge) continue;
if(!point[to].dfn){
pre[to] = u; ///如果下一个点没被访问过则更新一下下个点的pre
tarjan(to, i);
point[u].low = min(point[u].low, point[to].low);
if(point[to].low > point[u].dfn){
edge[i].bridge = true;
edge[i^].bridge = true;
bridge ++;
}
}else if(vis[to]){
point[u].low = min(point[to].dfn, point[u].low);
}
}
if(point[u].dfn == point[u].low){
vis[u] = false;
point[u].color = ++ colorNum;
while(stk.top() != u){
pre[stk.top()] = pre[u]; ///缩点时,该环中的所有点pre等于时间戳最小点的pre
point[stk.top()].color = colorNum;
vis[stk.top()] = false;
stk.pop();
}
stk.pop();
}
}
public:
void clear(int n){
sumOfPoint = n;
for(int i = ; i <= n; i ++){
first[i] = -;
pre[i] = -;
vis[i] = ;
point[i].dfn = ;
}
sign = colorNum = bridge = ;
dfnNum = ;
while(!stk.empty()) stk.pop();
}
void addEdgeOneWay(int u, int v){
edge[sign].to = v;
edge[sign].next = first[u];
edge[sign].bridge = false;
first[u] = sign ++;
}
void addEdgeTwoWay(int u, int v){
addEdgeOneWay(u, v);
addEdgeOneWay(v, u);
}
void tarjanAllPoint(){
for(int i = ; i <= sumOfPoint; i ++){
if(!point[i].dfn)
tarjan(i);
}
}
int getAns(int a, int b){
for(int i = ; i <= colorNum; i ++){
vis[i] = false;
}
vis[point[a].color] = true;
vis[point[b].color] = true;
int lca, lcacolor, ta = a, tb = b;
while(true){
if(ta != -) ta = pre[ta];
if(tb != -) tb = pre[tb];
if(vis[point[ta].color]){
lcacolor = point[ta].color;
lca = ta;
break;
}
if(vis[point[tb].color]){
lcacolor = point[tb].color;
lca = tb;
break; }
vis[point[ta].color] = true;
vis[point[tb].color] = true;
}
while(point[a].color != lcacolor){
for(int i = first[a]; i != -; i = edge[i].next){
int to = edge[i].to;
if(to == pre[a] && edge[i].bridge){
bridge --;
edge[i].bridge = false;
edge[i^].bridge = false;
break;
}
}
point[a].color = lcacolor;
int tmp = pre[a];
pre[a] = pre[lca];
a = tmp;
}
while(point[b].color != lcacolor){
for(int i = first[b]; i != -; i = edge[i].next){
int to = edge[i].to;
if(to == pre[b] && edge[i].bridge){
bridge --;
edge[i].bridge = false;
edge[i^].bridge = false;
break;
}
}
point[b].color = lcacolor;
int tmp = pre[b];
pre[b] = pre[lca];
b = tmp;
}
addEdgeTwoWay(a, b);
return bridge;
}
}graph; int main(){
int n, m, cas = ;
while(scanf("%d%d", &n, &m) != EOF && m + n){
graph.clear(n);
while(m --){
int a, b;
scanf("%d%d", &a, &b);
graph.addEdgeTwoWay(a, b);
}
graph.tarjanAllPoint();
int q;
scanf("%d", &q);
printf("Case %d:\n", cas ++);
while(q --){
int a, b;
scanf("%d%d", &a, &b);
printf("%d\n", graph.getAns(a, b));
}
putchar('\n');
}
return ;
}
poj 3694 Network : o(n) tarjan + O(n) lca + O(m) 维护 总复杂度 O(m*q)的更多相关文章
- POJ 3694 Network(无向图求桥+重边处理+LCA)
题目大意: 给你一个无向图,然后再给你一个Q代表有Q次询问,每一次加一条边之后还有几座桥.在这里要对重边进行处理. 每次加入一条边之后,在这条搜索树上两个点的公共祖先都上所有点的桥都没了. 这里重边的 ...
- Poj 3694 Network (连通图缩点+LCA+并查集)
题目链接: Poj 3694 Network 题目描述: 给出一个无向连通图,加入一系列边指定的后,问还剩下多少个桥? 解题思路: 先求出图的双连通分支,然后缩点重新建图,加入一个指定的边后,求出这条 ...
- POJ 3694——Network——————【连通图,LCA求桥】
Network Time Limit:5000MS Memory Limit:65536KB 64bit IO Format:%I64d & %I64u Submit Stat ...
- Poj 1236 Network of Schools (Tarjan)
题目链接: Poj 1236 Network of Schools 题目描述: 有n个学校,学校之间有一些单向的用来发射无线电的线路,当一个学校得到网络可以通过线路向其他学校传输网络,1:至少分配几个 ...
- POJ 3694 Network (tarjan + LCA)
题目链接:http://poj.org/problem?id=3694 题意是给你一个无向图n个点,m条边,将m条边连接起来之后形成一个图,有Q个询问,问将u和v连接起来后图中还有多少个桥. 首先用t ...
- POJ 3694 Network(Tarjan求割边+LCA)
Network Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 10969 Accepted: 4096 Descript ...
- poj 3694 Network 【Tarjan】+【LCA】
<题目链接> 题目大意: 给一个无向图,该图只有一个连通分量.然后查询q次,q < 1000, 求每次查询就增加一条边,求剩余桥的个数. 解题分析: 普通的做法就是在每加一条边后,都 ...
- poj 3694 Network(双连通分量)
题目:http://poj.org/problem?id=3694 #include <iostream> #include <cstring> #include <cs ...
- poj 3694 Network(割边+lca)
题目链接:http://poj.org/problem?id=3694 题意:一个无向图中本来有若干条桥,有Q个操作,每次加一条边(u,v),每次操作后输出桥的数目. 分析:通常的做法是:先求出该无向 ...
随机推荐
- 前端参数统一校验工具类ValidParamUtils
1,前端参数不可信,对于后端开发人员来说应该是一条铁律,所以对于前端参数的校验,必不可少,而统一的前端参数校验工具,对我们进行参数校验起到事半功倍的效果 2,统一参数校验工具ValidParamUti ...
- 软件项目技术点(2)——Canvas之坐标系转换
AxeSlide软件项目梳理 canvas绘图系列知识点整理 默认坐标系与当前坐标系 canvas中的坐标是从左上角开始的,x轴沿着水平方向(按像素)向右延伸,y轴沿垂直方向向下延伸.左上角坐标为 ...
- Ubuntu真机安装
Ubuntu真机安装 1.Ubuntu安装: (1)启动盘制作: a.下载启动盘制作工具Universal USB Installe,下载地址: b.下载Ubuntu系统镜像,到本地磁盘,官方下载地址 ...
- C++基础--指针,&的用法
#include "stdafx.h" #include <stdio.h> #include <string.h> int main() { ] = {, ...
- SELECT * FROM pet WHERE name REGEXP 'w';
To find names containing a “w”, use this query: SELECT * FROM pet WHERE name REGEXP 'w';
- java中什么是上下文(servletContext)
找了很多大佬的博客,看了之后还不是很清楚上下文到底是怎么回事,我个人理解 所谓上下文,它是用来存储系统的一些初始化信息,例如在jboss中通过配置文件指定了数据源,那么在jboss启动的时候就把这个文 ...
- SharePoint2010 HTTP Error 503. The service is unavailable 解决方法
1.更改系统管理员用户密码前提 因为更改系管理员用户密码会影响到 "SharePoint2010"正常运行,所在尽量不要更改系统管理员用户的密码, 必须更改密码的话,需要注意以两点 ...
- 屏幕 z
private void FullScreen() //全屏 { SizeMode = 2; FormBorderStyle = FormBo ...
- 《O2O实战:二维码全渠道营销》读书笔记思维导图(530KB)
- 使用 webpack 各种插件提升你的开发效率
前沿 项目地址 vue-admin 欢迎 star 近几个月,接手了一个老项目的重构规划,有多老呢?就是前端青铜时代的项目,一个前后端都在同一个锅里的项目.完全没有使用任何的打包工具. 后台 php ...