poj1966Cable TV Network(无向图最小点割集 ISAP+邻接矩阵)
邻接表的ISAP被卡了一天。。。TLE。。。。终于被卡了。。。好忧桑啊啊啊。。。
题目大意:给一张无向图,求最少去掉几个点使图不连通。
题目分析:求无向图的点连通度,拆点建图跑最大流。具体做法是:将一个点i拆成2个点:i和i+n,分别表示从第i个点出去和进入第i个点。那么i+n->i建边,边权1,对于每一条边(a,b),建边a->b + n,b->a+n,边权无穷。然后枚举没有边直接相连的点对(a,b),以a为源点,b+n为汇点跑最大流,最大流量就是该图的一个割,枚举所有不相邻点对,求出最小割。具体原理就是求不相邻点对(a,b)之间的最大独立轨数目,其实就是从a出发到达b的没有公共交点的路径数目。按照上述方式建图后,每个点i和i+n之间边权为1,保证每个点只存在某一条独立轨中,这样最大流的流量就是a到b的独立轨数量。
这题一开始熟悉的邻接表+ISAP做的,怎么交都是TLE了,没办法,只好改成邻接矩阵,竟然很快过了。这题复杂度还是很高的,说明数据不强,但是邻接表为什么就TLE了呢。。。
终于发现问题了。。。原来是输入函数导致的TLE。。。
详情请见代码:
邻接表+ISAP:
#include <iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N = 101;
const int M = 50005;
const int inf = 0x3f3f3f3f;
int n,m,num;
struct node
{
int to,next,c,pre;
}arc[M];
int head[N],que[N],sta[N],cnt[N],dis[N],rpath[N];
int st,ed;
bool map[N][N];
int d[M][2];
void build(int s,int e,int cap)
{
arc[num].to = e;
arc[num].c = cap;
arc[num].next = head[s];
head[s] = num ++;
arc[num - 1].pre = num;
arc[num].pre = num - 1;
arc[num].to = s;
arc[num].c = 0;
arc[num].next = head[e];
head[e] = num ++;
}
void re_Bfs()
{
int i,front,rear;
for(i = 0;i < n + n;i ++)
{
dis[i] = inf;
cnt[i] = 0;
}
front = rear = 0;
cnt[0] = 1;
dis[ed] = 0;
que[rear ++] = ed;
while(front != rear)
{
int u = que[front ++];
for(i = head[u];i != -1;i = arc[i].next)
{
if(arc[arc[i].pre].c == 0 || dis[arc[i].to] < inf)
continue;
dis[arc[i].to] = dis[u] + 1;
cnt[dis[arc[i].to]] ++;
que[rear ++] = arc[i].to;
}
}
}
int ISAP()
{
re_Bfs();
int i,u,v,ret = 0;
for(i = 0;i < n + n;i ++)
sta[i] = head[i];
u = st;
while(dis[st] < n + n)
{
if(u == ed)
{
int curflow = inf;
for(i = st;i != ed;i = arc[sta[i]].to)
curflow = min(curflow,arc[sta[i]].c);
for(i = st;i != ed;i = arc[sta[i]].to)
{
arc[sta[i]].c -= curflow;
arc[arc[sta[i]].pre].c += curflow;
}
ret += curflow;
u = st;
}
for(i = sta[u];i != -1;i = arc[i].next)
if(arc[i].c > 0 && dis[u] == dis[arc[i].to] + 1)
break;
if(i != -1)
{
sta[u] = i;
rpath[arc[i].to] = arc[i].pre;
u = arc[i].to;
}
else
{
if((-- cnt[dis[u]]) == 0)
break;
sta[u] = head[u];
int tmp = n + n + 1;
for(i = sta[u];i != -1;i = arc[i].next)
if(arc[i].c > 0)
tmp = min(tmp,dis[arc[i].to]);
dis[u] = tmp + 1;
cnt[dis[u]] ++;
if(u != st)
u = arc[rpath[u]].to;
}
}
return ret;
}
int nextint()
{
int ret;
char ch;
while((ch = getchar()) > '9' || ch < '0')
;
ret = ch - '0';
while((ch - getchar()) >= '0' && ch <= '9')
ret = ret * 10 + ch - '0';
return ret;
}
void buildgraph()
{
int i,j;
memset(head,-1,sizeof(head));
num = 0;
for(i = 0;i < n;i ++)
build(i + n,i,1);
for(i = 1;i <= m;i ++)
{
build(d[i][0],d[i][1] + n,inf);
build(d[i][1],d[i][0] + n,inf);
}
}
int main()
{
int i,j;
int a,b;
//freopen("in.txt","r",stdin);
while(scanf("%d",&n) != EOF)
{
scanf("%d",&m);
memset(map,false,sizeof(map));
i = 1;
while(m --)
{
scanf(" (%d,%d)",&a,&b);
//a = nextint();b = nextint();
if(map[a][b] == false)
{
map[a][b] = map[b][a] = true;
d[i][0] = a;
d[i ++][1] = b;
}
}
m = i - 1;
int ans = inf;
for(i = 0;i < n;i ++)
{
for(j = 0;j < i;j ++)
{
if(map[i][j] == false)
{
st = i;ed = j + n;
buildgraph();
ans = min(ans,ISAP());
}
}
}
if(ans == inf)
ans = n;
printf("%d\n",ans);
}
return 0;
}
//168K 16MS
为什么那个输入函数会导致TLE呢,只是想过滤掉字符而已。下面的代码也是这样用的啊啊 啊。路过的大神求科普!!
邻接矩阵+ISAP:
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N = 101;
const int inf = 0x3f3f3f3f;
int cap[N][N];
int flow[N][N];
bool flag[N][N];
int cnt[N],pre[N],dis[N],que[N];
int m,n,num,st,ed; void re_Bfs()
{
int front,rear,i;
for(i = 0;i < n + n;i ++)
dis[i] = inf,cnt[i] = 0;
dis[ed] = 0;
cnt[0] = 1;
front = rear = 0;
que[rear ++] = ed;
while(front != rear)
{
int u = que[front ++];
for(i = 0;i < n + n;i ++)
if(flow[i][u] < cap[i][u] && dis[i] > n + n)
{
dis[i] = dis[u] + 1;
cnt[dis[i]] ++;
que[rear ++] = i;
}
}
}
int ISAP()
{
int i,u,ret = 0;
memset(pre,-1,sizeof(pre));
memset(flow,0,sizeof(flow));
re_Bfs();
u = st;
while(dis[st] < n + n)
{
if(u == ed)
{
int tmp = inf;
for(i = ed;i != st;i = pre[i])
tmp = min(tmp,cap[pre[i]][i] - flow[pre[i]][i]);
for(i = ed;i != st;i = pre[i])
{
flow[pre[i]][i] += tmp;
flow[i][pre[i]] -= tmp;
}
ret += tmp;
u = st;
}
for(i = 0;i < n + n;i ++)
if(cap[u][i] > flow[u][i] && dis[u] == dis[i] + 1)
break;
if(i < n + n)
{
pre[i] = u;
u = i;
}
else
{
if((-- cnt[dis[u]]) == 0)
break;
int tmp = n + n;
for(i = 0;i < n + n;i ++)
if(cap[u][i] > flow[u][i] && dis[i] + 1 < tmp)
tmp = dis[i] + 1;
dis[u] = tmp;
cnt[tmp] ++;
if(pre[u] != st)
u = pre[u];
}
}
return ret;
}
int nextint()
{
int ret;
char c;
while((c = getchar()) > '9' || c < '0')
;
ret = c - '0';
while((c = getchar()) >= '0' && c <= '9')
ret = ret * 10 + c - '0';
return ret;
}
int main()
{
int i,j;
int a,b;
while(scanf("%d%d",&n,&m) != EOF)
{
memset(cap,0,sizeof(cap));
memset(flag,false,sizeof(flag));
for(i = 0;i < n;i ++)
cap[i + n][i] = 1;
for(i = 1;i <= m;i ++)
{
a = nextint();b = nextint();
cap[a][b + n] = inf;
cap[b][a + n] = inf;
flag[a][b] = flag[b][a] = true;
}
int ans = inf;
for(i = 0;i < n;i ++)
for(j = 0;j < i;j ++)
if(flag[i][j] == false)
{
st = i;ed = j + n;
ans = min(ans,ISAP());
}
if(ans == inf)
ans = n;
printf("%d\n",ans);
}
return 0;
}
//444K 47MS
poj1966Cable TV Network(无向图最小点割集 ISAP+邻接矩阵)的更多相关文章
- poj1966Cable TV Network——无向图最小割(最大流)
题目:http://poj.org/problem?id=1966 把一个点拆成入点和出点,之间连一条边权为1的边,跑最大流即最小割: 原始的边权赋成inf防割: 枚举源点和汇点,直接相邻的两个点不必 ...
- POJ 1966 Cable TV NETWORK(网络流-最小点割集)
Cable TV NETWORK The interconnection of the relays in a cable TV net ...
- POJ 1966:Cable TV Network(最小点割集)***
http://poj.org/problem?id=1966 题意:给出一个由n个点,m条边组成的无向图.求最少去掉多少点才能使得图中存在两点,它们之间不连通. 思路:将点i拆成a和b,连一条a-&g ...
- UVA-1660 Cable TV Network (最小割)
题目大意:给一张n个点.m条边的无向图,求最小点割集的基数. 题目分析:求无向图最小点割集的基数可以变成求最小割.考虑单源s单汇t的无向图,如果要求一个最小点集,使得去掉这个点集后图不再连通(连通分量 ...
- POJ 1966 Cable TV Network (无向图点连通度)
[题意]给出一个由n个点,m条边组成的无向图.求最少去掉多少点才能使得图中存在两点,它们之间不连通. [思路]回想一下s->t的最小点割,就是去掉多少个点能使得s.t不连通.那么求点连通度就枚举 ...
- POJ--1966--Cable TV Network【无向图顶点连通度】
链接:http://poj.org/problem?id=1966 题意:一个无向图,n个点,m条边,求此图的顶点连通度. 思路:顶点连通度,即最小割点集里的割点数目.一般求无向图顶点连通度的方法是转 ...
- POJ 1815 - Friendship - [拆点最大流求最小点割集][暴力枚举求升序割点] - [Dinic算法模板 - 邻接矩阵型]
妖怪题目,做到现在:2017/8/19 - 1:41…… 不过想想还是值得的,至少邻接矩阵型的Dinic算法模板get√ 题目链接:http://poj.org/problem?id=1815 Tim ...
- POJ 1966 Cable TV Network(顶点连通度的求解)
Cable TV Network Time Limit: 1000MS Memory Limit: 30000K Total Submissi ...
- UVA1660 电视网络 Cable TV Network
题目地址:UVA1660 电视网络 Cable TV Network 枚举两个不直接连通的点 \(S\) 和 \(T\) ,求在剩余的 \(n-2\) 个节点中最少去掉多少个可以使 \(S\) 和 \ ...
随机推荐
- SSIS 系列 - 在 SSIS 中使用 Multicast Task 将数据源数据同时写入多个目标表,备份数据表,以及写入Audit 信息
转自http://www.cnblogs.com/biwork/p/3328838.html 在 SSIS Data Flow 中有一个 Multicast 组件,它的作用和 Merge, Merge ...
- app行业发展趋势
近日,移动开放平台发布了2014年第一季度App开发行业报告.报告中对目前国内app开发者的分布情况,个人开发者和企业开发者的开发领域,相应比例以及提交应用过程中出现的问题做出统计,为如何建立一个更好 ...
- Bash Shell 快捷键的学习使用
原文地址: http://dbanotes.net/tech-memo/shell_shortcut.html 这篇 Bash Shell Shortcuts 的快捷键总结的非常好.值得学习.下面内容 ...
- 【转】【漫画解读】HDFS存储原理
根据Maneesh Varshney的漫画改编,以简洁易懂的漫画形式讲解HDFS存储机制与运行原理. 一.角色出演 如上图所示,HDFS存储相关角色与功能如下: Client:客户端,系统使用者,调用 ...
- 一笔画问题(floyd+oular+dfs)
一笔画问题 时间限制:3000 ms | 内存限制:65535 KB 难度:4 描述 zyc从小就比较喜欢玩一些小游戏,其中就包括画一笔画,他想请你帮他写一个程序,判断一个图是否能够用一笔画下 ...
- STL之nth_element()(取容器中的第n大值)
nth_element()函数 头文件:#include<algorithm> 作用:nth_element作用为求第n大的元素,并把它放在第n位置上,下标是从0開始计数的,也就是说求第0 ...
- minicom与USB转串口
实验器材:mini6410 连接方式:ARM板通过USB转串口线连接到pc机 下面是具体的设置了. 默认情况下,UBUNTU安装了USB转串口驱动(pl2303). 1.# lsmod | grep ...
- HTTP请求的TCP瓶颈分析[转]
阅读目录 延迟的因素 速度延时 带宽延时 最后一公里延时-tracerouter 目标 rwnd的设置 慢启动过程 慢启动的影响 慢启动对HTTP影响的一次计算 拥塞窗口的合适值 服务器配置调优 应用 ...
- Javascript进阶篇——( 事件响应)笔记整理
什么是事件 JavaScript 创建动态页面.事件是可以被 JavaScript 侦测到的行为. 网页中的每个元素都可以产生某些可以触发 JavaScript 函数或程序的事件. 鼠标单击事件(on ...
- 简单的javascript实例二(随页面滚动广告效果)
方便以后copy 页面代码: <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "htt ...