洛谷 P2055 [ZJOI2009]假期的宿舍 题解

【题目描述】

学校放假了......有些同学回家了,而有些同学则有以前的好朋友来探访,那么住宿就是一个问题。比如A和B都是学校的学生,A要回家,而C来看B,C与A不认识。我们假设每个人只能睡和自己直接认识的人的床。那么一个解决方案就是B睡A的床而C睡B的床。而实际情况可能非常复杂,有的人可能认识好多在校学生,在校学生之间也不一定都互相认识。我们已知一共有 nn 个人,并且知道其中每个人是不是本校学生,也知道每个本校学生是否回家。问是否存在一个方案使得所有不回家的本校学生和来看他们的其他人都有地方住。

【输入输出格式】

  • 输入格式

第一行一个数 TT 表示数据组数。接下来 TT 组数据,每组数据第一行一个数 nn 表示涉及到的总人数。接下来一行 nn个数,第 ii 个数表示第 ii 个人是否是在校学生( 00 表示不是, 11 表示是)。再接下来一行 nn 个数,第 ii 个数表示第 ii个人是否回家( 00 表示不回家, 11 表示回家,注意如果第 ii 个人不是在校学生,那么这个位置上的数是一个随机的数,你应该在读入以后忽略它)。接下来 nn 行每行 nn 个数,第 ii 行第 jj 个数表示 ii 和 jj 是否认识( 11 表示认识, 00 表示不认识,第 ii 行 ii 个的值为 00 ,但是显然自己还是可以睡自己的床),认识的关系是相互的。

  • 输出格式

对于每组数据,如果存在一个方案则输出“^_^”(不含引号)否则输出“T_T”(不含引号)。(注意输出的都是半角字符,即三个符号的 ASCII 码分别为94,84,95)。

【输入输出样例】

  • 输入样例
1
3
1 1 0
0 1 0
0 1 1
1 0 0
1 0 0
  • 输出样例
^_^

【数据范围】

对于 3030 %的数据满足 1 \leq n \leq 121≤n≤12 。

对于 100100 %的数据满足 1 \leq n \leq 501≤n≤50 , 1 \leq T \leq 201≤T≤20 。

思路:

这道题很明显就是一道二分图的最大匹配问题,当然这道题也可以用网络流来做,在讲完二分图匹配的思路后就会讲解最大流的思路了。你可以在这里检测你码的二分图最大匹配的模板的正确性。下面将详细地讲解二分图最大匹配这一算法。这里提出的二分图最大匹配问题与洛谷上的模板略有不同。如果你还没有理解什么是二分图匹配的话,那么就去学习下面的教程吧(教程转自《啊哈!算法》,放入题解时有删改)。


洛谷 P2055 [ZJOI2009]假期的宿舍 题解_第1张图片

教程:

小哼今天和小伙伴们一起去游乐场玩,终于可以坐上梦寐以求的过山车了。过山车的每一排只有两个座位,为了安全起见,每个女生必须与一个男生坐一排。但是,每个人都希望与自己认识的人坐在一起。举个例子吧, 11 好女生与 11 号男生互相认识,因此 11 号女生可以和 11 好男生坐在一起。另外 11 号女生与 22 号男生互相认识,因此他们也可以坐在一起。像这样的关系的还有 22 号女生和 22 号男生、 22 号女生和 33 号男生、 33 号女生和 11 号男生。请问如何安排座位才能让更多的人满意呢?这仅仅是个例子。实际情况要复杂得多,因为小哼的小伙伴们实在是太多了。

洛谷 P2055 [ZJOI2009]假期的宿舍 题解_第2张图片

首先我们先将这个问题模型化,如上图,左边的顶点是女生,右边的顶点是男生。如果顶点之间右边,就表示他们可以坐在一起,像这样特殊的图叫做二分图(注意二分图是无向图哦)。对于上面的例子,我们很容易找出两种分配方案,如下。

洛谷 P2055 [ZJOI2009]假期的宿舍 题解_第3张图片

  • 二分图的定义
  1. 如果一个图的所有的顶点都可以被分成 xx 和 yy 两个集合,并且所有边的两个顶点恰好属于一个集合 xx ,另一个属于集合 yy ,即每个集合内的顶点没有边相连,那么这张图就是二分图。

  2. 设 G=(V,E)G=(V,E) 是一个无向图,如果顶点V可分割为两个互不相交的子集 (A,B)(A,B) ,并且图中的每条边 (i,j)(i,j)所关联的两个顶点 ii 和 jj 分别属于这两个不同的顶点集( ii ∈ AA , jj ∈ BB ),则称图 GG 为一个二分图。

很显然,右边的分配方案更好。我们把一种分配方案叫做一种匹配。那么现在的问题就演变成了求二分图的最大匹配(匹配对数多)。求最大匹配容易想到的办法是:找出全部匹配,然后输出配对数最多的。这种方法的时间复杂度是非常之高的,那还有没有更好的办法呢? 我们可以这么想,首先先从左边的 11 号女生开始考虑。先让她与 11号男生配对,配对成功后就考虑 22 号女生与 22 号男生配对,接下来考虑 33 号女生。此时我们发现 33 号女生只能够跟 11 号男生配对,可是 11 号男生已经配对给了 11 号女生了,怎么办?其实我们可以让 33 号女生跟 11 号男生谈。让与 11 号男生相配对的 11 号女生一起谈,看看一号女生能不能够找到其他的同伴。如果能够找到,就让 33 号女生与 11号男生坐在一起,让 11 号女生与 11 号女生相配对的那个人坐在一起,若 11 号女生找到的其他的可以与 11 号女生配对的人已经有同伴的话就让他的同伴去找,看看其他人能否作为她的同伴,然后以此类推。若第 ii 名男生找到了同伴,就将答案加 11 。刚才的匹配过程就叫做增广路,不难发现,只要找到了一条增广路答案就会加 11 ,最后只需要输出答案即可。 下面给出两组数据用于检测您的程序的正确性(注: 11 到 33 号是男生, 44 到 66 号是女生):

  • 数据输入 11
6 5
1 4
1 5
2 5
2 6
3 4
  • 数据输出 11
3
  • 数据输入 22
6 4
1 4
2 5
2 6
3 4
  • 数据输出 22
3

下面就是二分图最大匹配的模板:

#include 
int book[10001];
int match[10001];
bool e[101][101];
int ans=0,n=0,m=0;
bool dfs(int u)
{
    for(int i=1;i<=n;i++)
    {
        if(book[i]==0 && e[u][i]==true)
        {
            book[i]=1;
            if(match[i]==0 || dfs(match[i])==true)
            {
                match[u]=i;
                match[i]=u;
                return true;
            }
        }
    }
    return false;
}
int main()
{
    scanf("%d %d",&n,&m);
    for(int i=1;i<=m;i++)
    {
        int x=0,y=0;
        scanf("%d %d",&x,&y);
        e[x][y]=true;
        e[y][x]=true;
    }
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=n;j++)
        {
            book[j]=0;
        }
        if(dfs(i)==true)
        {
            ans++;
        }
    }
    printf("%d",ans);
    return 0;
}

 


题目分析:

那么既然二分图最大匹配的思路讲完了,我们就要开始尝试去用二分图最大匹配来解决这道题目了。我们可以把人分给床。下面是几种情况:

  1. 如果他在学校住并且是本校学生,那么就让他睡回自己的床。

  2. 如果他在不在学校住或者他不回家,且他找不到自己的床。那么就返回 falsefalse , falsefalse 表示不存在一个方案使得所有不回家的本校学生和来看他们的其他人都有地方住,否则当返回 truetrue 时就表示存在一个方案使得所有不回家的本校学生和来看他们的其他人都有地方住。

  3. 如果他可以和这张床配对(即他认识这张床的主人),如果现在还没有人和这张床配对的话就直接让他和这张床配对,否则就看看现在与这张床配对的人可不可以找到另一张床可以和他配对,如果可以就让那个原本和那张床配对的人去和他找到的另一张可以与他配对的床相配对。再让这张现在没有配对的人与一开始提到的那个人相配对。

下面是一些注意事项:

  1. 是人认识床的主人,所以才能够睡在这张床上。所以连的是单向边,不是双向边。

  2. 要初始化,因为有多组数据。

讲完了二分图匹配的思路,下面就讲一讲网络流的思路吧,如下。你可以在这里检验你的网络流模板的正确性。下面将详细地讲解网络流这一算法。

对于下图,我们很容易就能够看出来这是一张单向图

洛谷 P2055 [ZJOI2009]假期的宿舍 题解_第4张图片

现在,我们给这几条边标上流量(也可以说是边权或者是最大流量)。注意,现在要标上流量的是边而不是点。什么是流量?我们不妨把从 xx 点到 yy 点的最大流量当做从 xx 点到 yy 点可以流过去的最大的水的重量,求从起始点处打开水龙头,让水通过这些“水管”(即边)流到到结束点处。现在我们的问题是,有 nn 个点 mm 条边,求源点到汇点的最大流量是多少。源点即起点,汇点即终点。在这里,我们的源点是 11 ,汇点是点 22 。

洛谷 P2055 [ZJOI2009]假期的宿舍 题解_第5张图片


网络流:

你会发现那些没有被标到层的都是无法到达的。于是我们便可以用广搜来判断能否到达汇点,若能够到达就计算有多少的流量能够流到汇点即可,当然,前提是还有水可以流过去。但是我们要注意一点:流过去的水如果留不到汇点还会流回来,因此我们对每一条单向边建一条流量为 00 的反向边即可,当有水流回来的时候加上反向边的流量即可。还有一点要注意的是,因为我们流水过去到汇点的次数可能会有多次,所以我们要加一个循环来完成网络流的整个过程。

下面上网络流的模板,你可以根据网络流的模板来进行进一步的学习,同时也感谢@liusu201601的建议。

#include 
#include 
#include 
struct nodea{ int x,y,d,g,f; } e[2000002];
struct nodeb{ int first,h; } r[1000001];
int q[1000001],len=0,m=0,n=0,st=0,ed=0;
int min(int x,int y)
{
    return x0;i=e[i].g)
        {
            int y=e[i].y;
            if(e[i].d>0 && r[y].h==0)
            {
                r[y].h=r[x].h+1;
                q[wei]=y;
                wei++;
            }
        }
        tou++;
    }
    if(r[ed].h>0)
    {
        return true;
    }
    return false;
}
int dfs(int x,int f)
{
    if(x==ed)
    {
        return f;
    }
    int tt=0;
    for(int i=r[x].first;i>0;i=e[i].g)
    {
        int y=e[i].y;
        if(r[x].h+1==r[y].h && tt<=f && e[i].d>0)
        {
            int my=dfs(y,min(e[i].d,f-tt));
            tt+=my;
            e[i].d-=my;
            e[e[i].f].d+=my;
        }
    }
    if(tt==0)
    {
        r[x].h=0;
    }
    return tt;
}
int main()
{
    scanf("%d %d %d %d",&n,&m,&st,&ed);
    for(int i=1;i<=m;i++)
    {
        int x=0,y=0,d=0;
        scanf("%d %d %d",&x,&y,&d);
        ins(x,y,d);
    }
    int ans=0;
    while(bfs()==true)
    {
        ans+=dfs(st,2099999999);
    }
    printf("%d",ans);
    return 0;
}

那么,对于网络流,该如何建图呢?

  1. 将人与源点连线,并将床与汇点连线。在这里,我用 ii 来表示第 ii 个人,用 i+ni+n 来表示第 ii 张床。

  2. 若一个人有床(即这个人是在下学生)就将这张床与汇点连线。若一个人需要床(即是本校学生的好朋友或是本校学生在这里住),就将其与源点相连线。若 ii 认识 jj ,那么 ii 就可以睡 jj 的床,那么就将 ii 和 jj 的床相连线,即将 ii 和 j+nj+n 相连线。

  3. 因为有多组数据,要注意将数组都初始化。一开始我就因为有两个数组忘了初始化造成了死循环,但是将这几组数据拆开后又是对的,这可能就是你的初始化没有初始化好了。一开始我就是因为没有初始化错了好几次,后来才通过的。



代码*2

下面上 100100 分代码~

二分图匹配:

#include 
#include 
int book[10001];
int match[10001];
int e[101][101];
int rn[10001],ho[10001];
int ans=0,n=0;
bool dfs(int u)
{
    for(int i=1;i<=n;i++)
    {
        if(book[i]==0 && rn[i]==1 && e[u][i]==1)
        {
            book[i]=1;
            if(match[i]==0 || dfs(match[i])==true)
            {
                match[i]=u;
                return true;
            }
        }
    }
    return false;
}
bool work()
{
    for(int i=1;i<=n;i++)
    {
        memset(book,0,sizeof(book));
        if((rn[i]==0 || ho[i]==0) && (dfs(i)==false))
        {
            return false;
        }
    }
    return true;
}
int main()
{
    int t=0;
    scanf("%d",&t);
    while(t--)
    {
        ans=0;
        memset(book,0,sizeof(book));
        memset(match,0,sizeof(match));
        memset(rn,0,sizeof(rn));
        memset(ho,0,sizeof(ho));
        memset(e,0,sizeof(e));
        scanf("%d",&n);
        for(int i=1;i<=n;i++)
        {
            scanf("%d",&rn[i]);
        }
        for(int i=1;i<=n;i++)
        {
            scanf("%d",&ho[i]);
            if(rn[i]==0)
            {
                ho[i]=1;
            }
        }
        for(int i=1;i<=n;i++)
        {
            for(int j=1;j<=n;j++)
            {
                scanf("%d",&e[i][j]);
            }
            if(rn[i]==1)
            {
                e[i][i]=1;
            }
        }
        if(work()==true)
        {
            printf("^_^\n");
        }
        else
        {
            printf("T_T\n");
        }
    }
    return 0;
}

网络流:

#include 
#include 
#include 
struct nodea{ int x,y,d,g,f; } e[1000001];
struct nodeb{ int first,h; } r[1000001];
int inf=999999999,st=1001,ed=1002,n=0;
int q[1000001],len=0;
int min(int x,int y)
{
    return x0;i=e[i].g)
        {
            int y=e[i].y;
            if(e[i].d>0 && r[y].h==0)
            {
                r[y].h=r[x].h+1;
                q[wei]=y;
                wei++;
            }
        }
        tou++;
    }
    if(r[ed].h>0)
    {
        return true;
    }
    return false;
}
int dfs(int x,int f)
{
    if(x==ed)
    {
        return f;
    }
    int tt=0;
    for(int i=r[x].first;i>0;i=e[i].g)
    {
        int y=e[i].y;
        if(r[x].h+1==r[y].h && tt<=f && e[i].d>0)
        {
            int my=dfs(y,min(e[i].d,f-tt));
            tt+=my;
            e[i].d-=my;
            e[e[i].f].d+=my;
        }
    }
    if(tt==0)
    {
        r[x].h=0;
    }
    return tt;
}
int dinic()
{
    int ans=0;
    while(bfs()==true)
    {
        ans+=dfs(st,inf);
    }
    return ans;
}
int main()
{
    int t=0;
    scanf("%d",&t);
    while(t--)
    {
        int tot=0;
        scanf("%d",&n);
        memset(q,0,sizeof(q));
        memset(e,0,sizeof(e));
        memset(r,0,sizeof(r));
        len=0;
        int rn[101],ho[101];
        for(int i=1;i<=n;i++)
        {
            scanf("%d",&rn[i]);
            if(rn[i]==1)
            {
                ins(i+n,ed,1);
            }
        }
        for(int i=1;i<=n;i++)
        {
            scanf("%d",&ho[i]);
            if((rn[i]==1 && ho[i]==0) || (rn[i]==0))
            {
                ins(st,i,1);
                tot++;
            }
        }
        for(int i=1;i<=n;i++)
        {
            for(int j=1;j<=n;j++)
            {
                int dx=0;
                scanf("%d",&dx);
                if(dx==1 || i==j)
                {
                    ins(i,j+n,1);
                }
            }
        }
        if(dinic()>=tot)
        {
            printf("^_^\n");
        }
        else
        {
            printf("T_T\n");
        }
    }
    return 0;
}

 



THE END(逃......

你可能感兴趣的:(题解)