题意:

  一些人,和他们可能加入的组号。问每个组,最小的最大人数是多少

分析:

  二分的是最大流的容量。设置一个超级源点,连向所有的人,容量为1。设置一个超级汇点,使所有的组连向超级汇点,二分的就是这里的容量(0~n)。然后根据题目给出的人和组的关系连接人和组,容量为1。二分时,若当前状态求出的最大流等于人数,则下界等于mid,若不等于,则上界等于mid。二分出的容量,就是组中的最小的最大人数。

输入:

3 2

John 0 1

Rose 1

Mary 1

5 4

ACM 1 2 3

ICPC 0 1

Asian 0 2 3

Regional 1 2

ShangHai 0 2

0 0

输出:

2

2

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#include <vector>
#include <sstream>
using namespace std;
const int maxn=3010;
const int inf=1e9;
int n,m;
structEdge
{
int u,v;
int flow,cap;
Edge(int uu,int cc,int fl,int ca):u(uu),v(cc),flow(fl),cap(ca)
{ }
};
vector<Edge>edge;
structISAP
{
vector<int>G[maxn];
vector<Edge>edge;
int s,t;
int d[maxn];
int cur[maxn];
int p[maxn];
int num[maxn];
int flow;
void init()
{
for(int i=0;i<=n+m+1;i++)
G[i].clear();
s=0;
t=n+m+1;
}
bool bfs()
{
memset(d,-1,sizeof(d));
queue<int>q;
q.push(t);
d[t]=0;
while(!q.empty())
{
int u=q.front();
q.pop();
for(int i=0;i<G[u].size();i++)
{
Edge& e=edge[G[u][i]];
if(e.cap==0&&d[e.v]==-1)
{
d[e.v]=d[u]+1;
q.push(e.v);
}
}
}
return d[s]!=-1;
}
int Augment()
{
int x=t,a=inf;
while(x!=s)
{
Edge& e=edge[p[x]];
a=min(a,e.cap-e.flow);
x=edge[p[x]].u;
}
x=t;
while(x!=s)
{
edge[p[x]].flow+=a;
edge[p[x]^1].flow-=a;
x=edge[p[x]].u;
}
return a;
}
int maxflow(int s,int t)
{
this->s=s;
this->t=t;
flow=0;
bfs();
for(int i=0;i<=n+m+1;i++)
num[d[i]]++;
int x=s;
memset(cur,0,sizeof(cur));
while(d[s]<n+m+2)
{
if(x==t)
{
flow+=Augment();
x=s;
}
bool ok=false;
for(int i=cur[x];i<G[x].size();i++)
{
Edge& e=edge[G[x][i]];
if(e.cap>e.flow&&d[x]==d[e.v]+1)
{
ok=true;
p[e.v]=G[x][i];
cur[x]=i;
x=e.v;
break;
}
}
if(!ok)
{
int k=n+m+1;
for(int i=0;i<G[x].size();i++)
{
Edge& e=edge[G[x][i]];
if(e.cap>e.flow)
k=min(k,d[e.v]);
}
if(--num[d[x]]==0)
break;
num[d[x]=k+1]++;
cur[x]=0;
if(x!=s)
x=edge[p[x]].u;
}
}
return flow;
}
}solver;
char buffer[11000];
void add(int s,int t,int cap)
{
edge.push_back(Edge(s,t,0,cap));
edge.push_back(Edge(t,s,0,0));
int x=edge.size();
solver.G[s].push_back(x-2);
solver.G[t].push_back(x-1);
}
void input()
{
solver.init();
int x;
edge.clear();
char ch[100];
for (int i = 1 ; i <= n ; ++i)
{
add(0,i,1);
scanf("%*s");
gets(buffer);
char * p = strtok(buffer, " ");
while (p) {
sscanf(p,"%d",&x);
add(i,x+1+n,1);
p = strtok(NULL," ");
}
}
int sz=edge.size();
for(int i=n+1;i<=n+m;i++)
{
sz+=2;
solver.G[i].push_back(sz-2);
solver.G[n+m+1].push_back(sz-1);
}
}
void build(int up)
{
solver.edge=edge;
for(int i=n+1;i<=n+m;i++)
{
solver.edge.push_back(Edge(i,n+m+1,0,up));
solver.edge.push_back(Edge(n+m+1,i,0,0));
}
}
void solve()
{
int left=1,right=n;
int mid=(left+right)>>1;
int ans=n;
while(left<=right)
{
build(mid);
if (solver.maxflow(0,n+m+1)==n)
{
right = mid-1;
ans = min(ans,mid);
}
else
{
left = mid+1;
}
mid = (left+right)>>1;
}
printf("%d\n",ans);
}
int main()
{
while(scanf("%d%d",&n,&m)==2,n+m)
{
input();
solve();
}
}
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#include <vector>
#include <sstream>
using namespace std;
const int maxn=3010;
const int inf=1e9;
int n,m;
structEdge
{
int u,v;
int flow,cap;
Edge(int uu,int cc,int fl,int ca):u(uu),v(cc),flow(fl),cap(ca)
{ }
};
vector<Edge>edge;
structISAP
{
vector<int>G[maxn];
vector<Edge>edge;
int s,t;
int d[maxn];
int cur[maxn];
int p[maxn];
int num[maxn];
int flow;
void init()
{
for(int i=0;i<=n+m+1;i++)
G[i].clear();
s=0;
t=n+m+1;
}
bool bfs()
{
memset(d,-1,sizeof(d));
queue<int>q;
q.push(t);
d[t]=0;
while(!q.empty())
{
int u=q.front();
q.pop();
for(int i=0;i<G[u].size();i++)
{
Edge& e=edge[G[u][i]];
if(e.cap==0&&d[e.v]==-1)
{
d[e.v]=d[u]+1;
q.push(e.v);
}
}
}
return d[s]!=-1;
}
int Augment()
{
int x=t,a=inf;
while(x!=s)
{
Edge& e=edge[p[x]];
a=min(a,e.cap-e.flow);
x=edge[p[x]].u;
}
x=t;
while(x!=s)
{
edge[p[x]].flow+=a;
edge[p[x]^1].flow-=a;
x=edge[p[x]].u;
}
return a;
}
int maxflow(int s,int t)
{
this->s=s;
this->t=t;
flow=0;
bfs();
for(int i=0;i<=n+m+1;i++)
num[d[i]]++;
int x=s;
memset(cur,0,sizeof(cur));
while(d[s]<n+m+2)
{
if(x==t)
{
flow+=Augment();
x=s;
}
bool ok=false;
for(int i=cur[x];i<G[x].size();i++)
{
Edge& e=edge[G[x][i]];
if(e.cap>e.flow&&d[x]==d[e.v]+1)
{
ok=true;
p[e.v]=G[x][i];
cur[x]=i;
x=e.v;
break;
}
}
if(!ok)
{
int k=n+m+1;
for(int i=0;i<G[x].size();i++)
{
Edge& e=edge[G[x][i]];
if(e.cap>e.flow)
k=min(k,d[e.v]);
}
if(--num[d[x]]==0)
break;
num[d[x]=k+1]++;
cur[x]=0;
if(x!=s)
x=edge[p[x]].u;
}
}
return flow;
}
}solver;
char buffer[11000];
void add(int s,int t,int cap)
{
edge.push_back(Edge(s,t,0,cap));
edge.push_back(Edge(t,s,0,0));
int x=edge.size();
solver.G[s].push_back(x-2);
solver.G[t].push_back(x-1);
}
void input()
{
solver.init();
int x;
edge.clear();
char ch[100];
for (int i = 1 ; i <= n ; ++i)
{
add(0,i,1);
scanf("%*s");
gets(buffer);
char * p = strtok(buffer, " ");
while (p) {
sscanf(p,"%d",&x);
add(i,x+1+n,1);
p = strtok(NULL," ");
}
}
int sz=edge.size();
for(int i=n+1;i<=n+m;i++)
{
sz+=2;
solver.G[i].push_back(sz-2);
solver.G[n+m+1].push_back(sz-1);
}
}
void build(int up)
{
solver.edge=edge;
for(int i=n+1;i<=n+m;i++)
{
solver.edge.push_back(Edge(i,n+m+1,0,up));
solver.edge.push_back(Edge(n+m+1,i,0,0));
}
}
void solve()
{
int left=1,right=n;
int mid=(left+right)>>1;
int ans=n;
while(left<=right)
{
build(mid);
if (solver.maxflow(0,n+m+1)==n)
{
right = mid-1;
ans = min(ans,mid);
}
else
{
left = mid+1;
}
mid = (left+right)>>1;
}
printf("%d\n",ans);
}
int main()
{
while(scanf("%d%d",&n,&m)==2,n+m)
{
input();
solve();
}
}

UVA 1345 Jamie's Contact Groups的更多相关文章

  1. POJ 2289 Jamie's Contact Groups / UVA 1345 Jamie's Contact Groups / ZOJ 2399 Jamie's Contact Groups / HDU 1699 Jamie's Contact Groups / SCU 1996 Jamie's Contact Groups (二分,二分图匹配)

    POJ 2289 Jamie's Contact Groups / UVA 1345 Jamie's Contact Groups / ZOJ 2399 Jamie's Contact Groups ...

  2. POJ2289 Jamie's Contact Groups(二分图多重匹配)

    Jamie's Contact Groups Time Limit: 7000MS   Memory Limit: 65536K Total Submissions: 7721   Accepted: ...

  3. Jamie's Contact Groups POJ - 2289(多重匹配 最大值最小化 最大流)

    Jamie's Contact Groups Time Limit: 7000MS   Memory Limit: 65536K Total Submissions: 8567   Accepted: ...

  4. POJ 2289 Jamie's Contact Groups 二分图多重匹配 难度:1

    Jamie's Contact Groups Time Limit: 7000MS   Memory Limit: 65536K Total Submissions: 6511   Accepted: ...

  5. poj 2289 Jamie's Contact Groups【二分+最大流】【二分图多重匹配问题】

    题目链接:http://poj.org/problem?id=2289 Jamie's Contact Groups Time Limit: 7000MS   Memory Limit: 65536K ...

  6. POJ2289:Jamie's Contact Groups(二分+二分图多重匹配)

    Jamie's Contact Groups Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 125536/65536 K (Java/ ...

  7. POJ 2289——Jamie's Contact Groups——————【多重匹配、二分枚举匹配次数】

    Jamie's Contact Groups Time Limit:7000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I ...

  8. POJ2289 Jamie's Contact Groups —— 二分图多重匹配/最大流 + 二分

    题目链接:https://vjudge.net/problem/POJ-2289 Jamie's Contact Groups Time Limit: 7000MS   Memory Limit: 6 ...

  9. Poj 2289 Jamie's Contact Groups (二分+二分图多重匹配)

    题目链接: Poj 2289 Jamie's Contact Groups 题目描述: 给出n个人的名单和每个人可以被分到的组,问将n个人分到m个组内,并且人数最多的组人数要尽量少,问人数最多的组有多 ...

随机推荐

  1. 关于select元素的一些基本知识

    为select元素绑定值的几个方法: 一.通过字符串拼接,让后追加到select元素下, 二.通过DOM创建option元素,为其绑上value值和文本: function loadProvinve( ...

  2. css解决文字垂直居中

    参考链接: http://www.cnblogs.com/lufy/archive/2012/09/12/2681972.html   http://zhidao.baidu.com/question ...

  3. JS实现下拉框选中不同的项,对应显示不同的信息

    实现的效果如下图: 页面代码 下拉框: <select id="select3" name="select3" onchange="showli ...

  4. java final 和instanceof 关键字

    /* final class A { public final void eat(){ System.out.println("测试"); } } */ class A { } c ...

  5. parseInt引发的血案

    今天做了个专题活动,页面头上有个倒计时 专题做完后上线了,没发现有什么问题,结果,运营MM突然和我说:技术哥哥出问题了,360浏览器在秒数从10到09的时候直接变成 00 了! 一看我去真的,该死的3 ...

  6. VS 2003 无法打开Web项目,位于服务器“http:10.45.4.70:8080”上的项目不存在

    解决方法: 用记事本打开*.sln文件更改第2行 改成正确的虚拟目录 出现这种情况往往是从一台机器搬到另一台机器造成的虚拟路径名字不同

  7. 常用网站--前端开发类+网页设计类+平面素材类+flash类

    前端开发类 animate CSS 前端开发网 我爱CSS 大家网 W3School jQuery开发技术详解教程视频 jQuery中文社区 jQueryChina 网页设计类 禅意花园 CSS Do ...

  8. css派生选择器

    后代选择器:即包含选择器,选择某元素的后代元素. 子元素选择器:只能选择某元素的子元素. 相邻兄弟选择器:可选择紧接在另一个元素后的元素,且两者有相同的夫元素.

  9. 如何解决安卓SDK无法下载Package的问题

    转载自:http://jingyan.baidu.com/article/8275fc86dbe84046a03cf69d.html 有些用户在安装好Android SDK后,打开Android SD ...

  10. Asp.Net通过SignalR实现IM即时通讯

    前言:SignalR是一种针对H5中WebSocket的解决方案,可以实现在不支持H5的浏览器中实现IM 后端: step 1:通过NuGet安装SignalR step 2:新建一个类继承于Hub, ...