poj 1815 Friendship (最小割+拆点+枚举)

题意:

就在一个给定的无向图中至少应该去掉几个顶点才能使得s和t不联通。

算法:

如果s和t直接相连输出no answer。

把每个点拆成两个点v和v‘‘,这两个点之间连一条权值为1的边(残余容量)

v和v‘‘分别是一个流进的点,一个流出的点。

根据求最小割的性质,权值小的边是可能被选择的(断开的)。

添加源点st=0和汇点en=2*n+1,源点与s连权值为inf的边,t‘‘与汇点连权值为inf的边。

s与s‘‘,t与t‘‘连权值为inf的边,这样保证自己和自己是不会失去联系的。

如果i和j有边相连,则i‘‘和j连权值为inf的边,j‘‘与i连权值为inf的边。

这样建图后跑最大流,求得的流量即为点的个数。

然后编号从小到大枚举每个点,尝试去掉这个点(即只进不出),重新建图再跑最大流,

看最大流是否会减小,如果减小了,就是要去掉的点。记录下来最后输出就可以了。

PS:

建图也可以不加源点和汇点,直接没去掉的点,拆的两点直接连权值为1的边,有边相连的

两点连权值为INF的边。

终于理解了我写的Dinic模板一直是直接处理残余网络即e[i].val的。还有把容量网络和流量分开写的Dinic。

#include<cstdio>
#include<iostream>
#include<cstring>
#define INF 0x3f3f3f3f
#define maxn 210
#define maxm 160000
using namespace std;

struct node
{
    int v,val,next;
}e[maxm<<1];
int head[maxn<<1],mp[maxn][maxn],cnt,st,en,s,t,n;
int d[maxn<<1],q[maxn<<1],mm[maxn],del[maxn];

void init()
{
    memset(del,0,sizeof(del));
    memset(mp,0,sizeof(mp));
    st = 0;
    en = 2*n+1;
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=n;j++)
            scanf("%d",&mp[i][j]);
    }
}

void add(int x,int y,int z)
{
    e[cnt].v = y;
    e[cnt].val = z;
    e[cnt].next = head[x];
    head[x] = cnt++;
    e[cnt].v = x;
    e[cnt].val = 0;
    e[cnt].next = head[y];
    head[y] = cnt++;
}

void build()
{
    memset(head,-1,sizeof(head));
    cnt = 0;
    add(st,s,INF);
    add(t+n,en,INF);
    for(int i=1;i<=n;i++)
    {
        if(!del[i]) add(i,i+n,1);
        for(int j=1;j<=n;j++)
        {
            if(mp[i][j])
                add(i+n,j,INF);
        }
    }
    add(s,s+n,INF);
    add(t,t+n,INF);
}
bool bfs()
{
    memset(d,-1,sizeof(d));
    int f = 0,r = 0,u;
    q[r++] = st;
    d[st] = 0;
    while(f<r)
    {
        u = q[f++];
        for(int i=head[u];i!=-1;i=e[i].next)
        {
            int t = e[i].v;
            if(e[i].val>0 && d[t]==-1)//>0
            {
                d[t] = d[u]+1;
                q[r++] = t;
                if(t==en) return true;
            }
        }
    }
    return false;
}

int dfs(int x,int flow)
{
    if(x==en) return flow;
    int ret = 0,dd;
    for(int i=head[x];ret<flow && i!=-1;i=e[i].next)
    {
        int t = e[i].v;
        if(d[t] == d[x]+1 && e[i].val)
        {
            dd = dfs(t,min(flow,e[i].val));
            e[i].val-=dd;
            e[i^1].val+=dd;
            flow-=dd;
            ret+=dd;
        }
    }
    if(!ret) d[x]=-1;
    return ret;
}
int Dinic()
{
    int tmp = 0,maxflow = 0;
    while(bfs())
    {
        while(tmp=dfs(st,INF))
            maxflow+=tmp;
    }
    return maxflow;
}

void solve()
{
    if(mp[s][t])
    {
        printf("NO ANSWER!\n");
        return;
    }
    build();
    int ans = Dinic();
    printf("%d\n",ans);
    if(!ans) return;
    int tmp = ans,f = 0,now;
    for(int i=1;i<=n;i++)
    {
        if(i==s || i==t) continue;
        //if(!mp[s][i]) continue;  //点i虽然与s不是直接连通,但可能间接连通,所以枚举时不能continue掉
        del[i] = 1;
        build();
        now = Dinic();
        if(now<tmp)
        {
            mm[f++] = i;
            tmp = now;
        }
        else
            del[i] = 0;
    }
    for(int i=0;i<f-1;i++)
        printf("%d ",mm[i]);
    printf("%d\n",mm[f-1]);
}
int main()
{
    while(scanf("%d%d%d",&n,&s,&t)!=EOF)
    {
        init();
        solve();
    }
    return 0;
}

/*

9 1 9
1 1 1 0 0 0 0 0 0
1 1 1 1 1 0 0 0 0
1 1 1 0 1 1 0 0 0
0 1 0 1 0 0 1 0 0
0 1 1 0 1 0 1 1 0
0 0 1 0 0 1 0 1 0
0 0 0 1 1 0 1 1 1
0 0 0 0 1 1 1 1 1
0 0 0 0 0 0 1 1 1

*/

poj 1815 Friendship (最小割+拆点+枚举)

时间: 2024-10-14 00:44:46

poj 1815 Friendship (最小割+拆点+枚举)的相关文章

POJ 1815 Friendship(最小割)

http://poj.org/problem?id=1815 Friendship Time Limit: 2000MS   Memory Limit: 20000K Total Submissions: 9026   Accepted: 2534 Description In modern society, each person has his own friends. Since all the people are very busy, they communicate with eac

poj 1815 Friendship 最小割输出最小方案

这题卡了好久啊,最小割模型很容易想,拆点就行.就像poj的Thieves一样 每个点 a拆成 a->a',容量为1. 其他相连的点 a'->b ,容量为INF 源连接s',t连接汇 问题在于输出最小的割集 更好的方法我还不会,只能枚举. 这里基于贪心的思想,从小到大删边, 若删除i->i',会使得最小割变小,则输出i,并且i->i'这条边不要恢复 若最小割不变,则恢复这条边,继续枚举. 一开始就是因为恢复了要割去的边,无限WA. #include<cstdio> #in

Poj 1815 Friendship 枚举+求最小割

给以一个图和两个点S,T,问你拿掉最少多少个点可以使得S和T不连通.输出点数并且输出拿掉的是哪些点,如果有多种方法就输出字典序最小的那个. 这就是一个求最小点割集的问题.无向(有向)图G中,给定源点s和终点t,至少要删去多少个点(具体一点,删哪些点),使得s和t不连通.这个问题就是点连通度,也叫最小点割集. 解法其实理解起来不难,只要把图中的每一个点v拆成v',v''两个点,并建立<v',v''>权为1,这样就把最小点割集转化成求最小割的问题. 对于原图的转化也很简单,对于原来的每条边,转化成

POJ 1815 Friendship(最小割)

POJ 1815 Friendship 链接:http://poj.org/problem?id=1815 题目:在现代社会,每个人都有自己的朋友.由于每个人都很忙,他们只通过电话联系.你可以假定A 可以和B 保持联系,当且仅当: (1) A 知道B 的电话号码,或 (2) A 知道C 的号码,而C 能联系上B. 如果A 知道B 的电话号码,则B 也知道A 的电话号码.有时,有人可能会碰到比较糟糕的事情,导致他与其他人失去联系.例如,他可能会丢失了电话簿,或者换了电话号码. 在本题中,告知N 个

zoj 2874 &amp; poj 3308 Paratroopers (最小割)

题意: 一个m*n大小的网格,已知伞兵着陆的具体位置(行和列).现在在某行(或某列) 安装一架激光枪,一架激光枪能杀死该行(或该列)所有的伞兵.在第i行安装一架 激光枪的费用是Ri,在第i列安装的费用是Ci.要安装整个激光枪系统,总费用为这些 激光枪费用的乘积. 求杀死所有伞兵的最小费用. 构图: 把伞兵视为边,行与列视为顶点.增加源点和汇点,对于第i行,从源点向顶点i连接一条 容量为Ri的边.对于第j列,从顶点j向汇点连接一条容量为Rj的边. 如果某一点(i,j)有伞兵降落,则从顶点Ri向顶点

POJ 1815 Friendship(最小割+字典序输出割点)

http://poj.org/problem?id=1815 题意: 在现代社会,每个人都有自己的朋友.由于每个人都很忙,他们只通过电话联系.你可以假定A可以和B保持联系,当且仅当:①A知道B的电话号码:②A知道C的电话号码,而C能联系上B.如果A知道B的电话号码,则B也知道A的电话号码. 思路:这题是要我们删点,既然是删点,那么就要拆点,容量就是1. 接下来凡是能联系的,就连边,容量为INF,因为我们不是要删除这些边.跑遍最大流就能算出至少要删除多少个点. 这道题的关键是要字典序顺序输出最小割

poj 1815 Friendship 字典序最小+最小割

题目链接:http://poj.org/problem?id=1815 In modern society, each person has his own friends. Since all the people are very busy, they communicate with each other only by phone. You can assume that people A can keep in touch with people B, only if 1. A kno

POJ 1815 Friendship

Friendship Time Limit: 2000MS   Memory Limit: 20000K Total Submissions: 10614   Accepted: 2945 Description In modern society, each person has his own friends. Since all the people are very busy, they communicate with each other only by phone. You can

poj 3308 Paratroopers 最小割 最小点权覆盖

题目链接:http://poj.org/problem?id=3308 题意: 有一个M*N的图,上面的一些点上有伞兵. 可以设置一些枪在每行或者每列上,通过射击,这行或这列的伞兵就会被消灭.每个枪的设置有一个花费,如果设置多个枪,那么花费是设置每个枪的乘积. 问消灭所有伞兵最少的花费是多少. 思路: 每个点的伞兵至少要用那一列或者那一行设置的枪去消灭,那么就可以应用点覆盖的模型.把伞兵看成是一条边,这条边至少要用一个点来覆盖. 而题目中最终花费是所有花费的乘积,那么可以用对数log(x)+lo