poj1966Cable TV Network(无向图最小点割集 ISAP+邻接矩阵)

摘要:
请在这里标记邻接表的ISAP已经卡住了一天……具体的原理是找到非相邻点对(a,b)之间独立轨道的最大数量,这实际上是从a到b没有公共交集的路径的数量。一开始,熟悉的邻接列表+ISAP解决了这个问题。然而,这是TLE。我别无选择,只能将其更改为邻接矩阵,这很快就通过了。有关详细信息,请参阅代码:邻接表+ISAP:#include#include#include#includeusingspacestd;组成N=101;组成M=50005;组成f=0x3f3f3f;整数,m,num;结构节点{intto,next,c,pre;}弧[M];在头[N],que[N],sta[N],cnt[N],dis[N],rpath[N];intst,ed;布尔映射[N][N];intd[M][2];voidbuild{arc[num].to=e;arc[num].c=cap;arc[num].next=head[s];head[s]=num++;arc[nnum-1].pre=num;arc[nnum].pre=num-1;arc[nm].to=s;arc[number].c=0;arc[n].next=head[e];head[e]=num++;}voire_Bfs(){inti,前,后;对于{dis[i]=inf;cnt[i]=0;}前=后=0;cnt[0]=1;dis[ed]=0;que[后++]=ed;while(front!=EOF){scanf;memset;i=1;while(m-){canf;//a=nextin();b=nextint();如果{map[a][b]=map[b][a]=true;d[i][0]=a;d[i++][1]=b;}}m=i-1;无形资产=inf;对于{如果{st=i;ed=j+n;buildgraph();ans=min;}}ifans=n;printf;}return0;}//168K16MS为什么输入功能会导致TLE?它只是想过滤掉字符。

题目请戳这里

邻接表的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
",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
",ans);
    }
    return 0;
}
//444K	47MS


免责声明:文章转载自《poj1966Cable TV Network(无向图最小点割集 ISAP+邻接矩阵)》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇TP手册学习第三天自学Zabbix3.4-资产清单inventory下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

整体二分

关于整体二分的学习其实只是了解了普通二分之后自然就懂了整体二分了。没有很难但是基于整体二分是基于时间和值域的分治。 比CDQ要更强一点。因为可以在一些离线的题目当中代替树套树什么的。 同时也非常的好写 。至少代码好懂。 首先是值域上的二分 这个二分是分离不同的值毕竟不同的值在不同的区间之内。我们可以将其二分。但是单次二分也是可以得到整体的信息的不妨把所有的...

Asp.Net 高性能框架 SqlSugar.ORM 2.3

一、前言SqlSugar从去年到现在已经一年了,版本从1.0升到了现在的2.3 ,这是一个稳定版本 ,有数家公司已经项目上线,在这里我将SqlSugar的功能重新整理成一篇新的贴子,希望大家喜欢。 公司团队项目、产品已经完全抛弃EF,SqlSugar定位不是ORM,而是为了方面的让你去写Sql。 支持Json 、Dynamic、 List<T>...

右键快捷菜单的创建及控件的动态添加

任务:熟悉c#的winform右键快捷菜单的使用,熟悉使用右键实现控件的动态添加 技术要点: 1.实现控件的动态添加,设置控件的位置、大小和名称等属性 2.熟悉ContextMenuStrip控件的使用 实现过程: 1.创建Winform窗体,并添加ContextMenuStrip控件 2.编辑ContextMenuStrip控件,为其添加“添加控件”和...

AVR单片机教程——走向高层

本文隶属于AVR单片机教程系列。   在系列教程的最后一篇中,我将向你推荐3个可以深造的方向:C++、事件驱动、RTOS。掌握这些技术可以帮助你更快、更好地开发更大的项目。 本文涉及到许多概念性的内容,如果你有不同意见,欢迎讨论。 关于高层 这一篇教程叫作“走向高层”。什么是高层? 我认为,如果寥寥几行代码就能实现一个复杂功能,或者一行代码可以对应到几百句...

6Java控制结构

顺序结构 从上到下依次执行,中间没有判断和跳转 比如Java中定义变量中的前向应用,先定义,在使用 分支结构(if, else, switch) 单分支 if // if 的快速入门 import java.util.Scanner; class if01{ public static void main(String[] args) {...

dpdk 网卡顺序

1、网卡扫描总线的func slot/probe函数的调用是有顺序的, 先绑定0000:05:00.0 ,后绑定0000:06:00.0  [root@localhost dpdk-19.11]# ./usertools/dpdk-devbind.py -s Network devices using DPDK-compatible driver=====...