[洛谷 3254]圆桌问题

题目链接

https://www.luogu.com.cn/problem/P3254

题解

容易看出这是一个二分图的模型。左部点为单位,右部点为餐桌。

我们设 \(i\) 单位的点为 \(p_i\),\(j\) 餐桌的点为 \(q_j\),按照下面的方式来建模:

  1. \(s \to p_i\),流量为 \(r_i\)。
  2. \(p_i \to q_j\),流量为 \(1\)(两个同单位的人不能在一张餐桌上)。
  3. \(q_j \to t\),流量为 \(c_i\)。

跑最大流。如果不满流则无解,否则方案为所有 \(p_i \to q_j\) 满流的边。

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#define INF 0x3f3f3f3f
using namespace std;
struct edge
{
 int v,w,next;
}e[100005];
int s,t,cnt=1;
int head[1005],dep[1005],vis[1005],cur[1005],nxt[1005];
void addedge(int u,int v,int w)
{
 e[++cnt].v=v;
 e[cnt].w=w;
 e[cnt].next=head[u];
 head[u]=cnt;
}
bool bfs()
{
 queue<int> q;
 memset(dep,INF,sizeof(dep));
 memset(vis,0,sizeof(vis));
 memcpy(cur,head,sizeof(head));
 dep[s]=0;
 vis[s]=1;
 q.push(s);
 while(!q.empty())
 {
  int p=q.front();
  q.pop();
  vis[p]=0;
  for(int i=head[p];i;i=e[i].next)
   if(dep[e[i].v]>dep[p]+1&&e[i].w)
   {
    dep[e[i].v]=dep[p]+1;
    if(!vis[e[i].v])
    {
     vis[e[i].v]=1;
     q.push(e[i].v);
    }
   }
 }
 if(dep[t]==INF)return 0;
 return 1;
}
int dfs(int p,int w)
{
 if(p==t)return w;
 int used=0;
 for(int i=cur[p];i;i=e[i].next)
 {
  cur[p]=i;
  if(dep[e[i].v]==dep[p]+1&&e[i].w)
  {
   int flow=dfs(e[i].v,min(w-used,e[i].w));
   if(flow)
   {
    used+=flow;
    e[i].w-=flow;
    e[i^1].w+=flow;
    if(used==w)break;
   }
  }
 }
 return used;
}
int main()
{
 int m,n,sum=0,ans=0;
 scanf("%d%d",&m,&n);
 s=m+n+1,t=m+n+2;
 for(int i=1;i<=m;i++)
 {
  int x;
  scanf("%d",&x);
  sum+=x;
  addedge(s,i,x);
  addedge(i,s,0);
  for(int j=n;j;j--)
  {
   addedge(i,m+j,1);
   addedge(m+j,i,0);
  }
 }
 for(int i=1;i<=n;i++)
 {
  int x;
  scanf("%d",&x);
  addedge(m+i,t,x);
  addedge(t,m+i,0);
 }
 while(bfs())
  ans+=dfs(s,INF);
 if(ans!=sum)puts("0");
 else
 {
  puts("1");
  for(int u=1;u<=m;u++)
  {
   for(int i=head[u];i;i=e[i].next)
   {
    if(e[i].v==s)continue;
    for(int j=1;j<=e[i^1].w;j++)
     printf("%d ",e[i].v-m);
   }
   puts("");
  }
 }
 return 0;
}

发表回复

您的电子邮箱地址不会被公开。 必填项已用 * 标注

此站点使用Akismet来减少垃圾评论。了解我们如何处理您的评论数据