1305: [CQOI2009]dance跳舞

Time Limit: 5 Sec  Memory Limit: 162 MB
Submit: 4169  Solved: 1804
[Submit][Status][Discuss]

Description

一次舞会有n个男孩和n个女孩。每首曲子开始时,所有男孩和女孩恰好配成n对跳交谊舞。每个男孩都不会和同一个女孩跳两首(或更多)舞曲。有一些男孩女孩相互喜欢,而其他相互不喜欢(不会“单向喜欢”)。每个男孩最多只愿意和k个不喜欢的女孩跳舞,而每个女孩也最多只愿意和k个不喜欢的男孩跳舞。给出每对男孩女孩是否相互喜欢的信息,舞会最多能有几首舞曲?

Input

第一行包含两个整数n和k。以下n行每行包含n个字符,其中第i行第j个字符为‘Y‘当且仅当男孩i和女孩j相互喜欢。

Output

仅一个数,即舞曲数目的最大值。

Sample Input

3 0
YYY
YYY
YYY

Sample Output

3

HINT

N<=50 K<=30

Source

加强数据By dwellings and liyizhen2

乍一看以为多重二分图匹配,花了半个小时写出来82分WA了,搜一遍题解才知道是网络流最大流拆点

将所有人拆为两个点,男生为xi,xj,女生为yi,yj

将相互喜欢的人,由xi连向yi,容量为1

将互相不喜欢的人,由xj连向yj,容量为1

将所有男生的xi连向xj,容量为k

将所有女生yj连向yi,容量为k

将源点连向xi,容量为a

  yi连向汇点,容量为a

二分法枚举容量a,计算最大流,若flow<n*a,即无法满流,则停止二分

  1 #include<iostream>
  2 #include<cstdio>
  3 #include<cstring>
  4 #include<queue>
  5 using namespace std;
  6
  7 const int INF=0x7f7f7f7f;
  8 const int MAXN=400000;
  9
 10 struct Edge
 11 {
 12     int to,w,next;
 13 }E[MAXN];
 14 int node=1,head[MAXN],dis[MAXN];
 15 int s=0,t=1000;
 16 int n,k,ans;
 17 bool mp[100][100];
 18
 19 void insert(int u,int v,int w)
 20 {
 21     E[++node]=(Edge){v,w,head[u]};
 22     head[u]=node;
 23     E[++node]=(Edge){u,0,head[v]};
 24     head[v]=node;
 25 }
 26
 27 bool bfs()
 28 {
 29     memset(dis,-1,sizeof(dis));
 30     queue<int> Q;
 31     Q.push(s);
 32     dis[s]=0;
 33     while(!Q.empty())
 34     {
 35         int q=Q.front();Q.pop();
 36         for(int i=head[q];i;i=E[i].next)
 37             if(E[i].w&&dis[E[i].to]==-1)
 38             {
 39                 Q.push(E[i].to);
 40                 dis[E[i].to]=dis[q]+1;
 41             }
 42     }
 43     return dis[t]!=-1;
 44 }
 45
 46 int dfs(int x,int flow)
 47 {
 48     if(x==t) return flow;
 49     int w,used=0;
 50     for(int i=head[x];i;i=E[i].next)
 51         if(E[i].w&&dis[E[i].to]==dis[x]+1)
 52         {
 53             w=flow-used;
 54             w=dfs(E[i].to,min(w,E[i].w));
 55             E[i].w-=w;
 56             E[i^1].w+=w;
 57             used+=w;
 58             if(used==flow)return flow;
 59         }
 60     if(!used) dis[x]=-1;
 61     return used;
 62 }
 63
 64 void dinic()
 65 {
 66     while(bfs()) ans+=dfs(s,INF);
 67 }
 68
 69 int main()
 70 {
 71     scanf("%d%d",&n,&k);
 72     for(int i=1;i<=n;i++)
 73     {
 74         char ch[100];
 75         scanf("%s",ch);
 76         for(int j=1;j<=n;j++)
 77             if(ch[j-1]==‘Y‘) mp[i][j]=1;
 78     }
 79     int left=0,right=50;
 80     while(left<=right)
 81     {
 82         int mid=(left+right)>>1;
 83         node=1;memset(head,0,sizeof(head));
 84         for(int i=1;i<=n;i++)
 85         {
 86             insert(0,i,mid);
 87             insert(i,i+500,k);
 88             insert(n+i+500,n+i,k);
 89             insert(n+i,t,mid);
 90             for(int j=1;j<=n;j++)
 91                 if(mp[i][j]) insert(i,j+n,1);
 92                 else insert(i+500,j+n+500,1);
 93         }
 94         ans=0;dinic();
 95         if(ans>=n*mid) left=mid+1;
 96         else right=mid-1;
 97     }
 98     printf("%d",left-1);
 99     return 0;
100 }

原文地址:https://www.cnblogs.com/InWILL/p/9733341.html

时间: 2024-10-14 20:56:26

1305: [CQOI2009]dance跳舞的相关文章

BZOJ 1305: [CQOI2009]dance跳舞( 最大流 )

云神代码很短...0 ms过的...看了代码 , 大概是贪心... orz 我不会证 数据这么小乱搞就可以了吧... ←_← 这道题网络流还是可以写的... 既然限制了最多只能和 k 个不喜欢的人dance ,  那么就把每个人拆成 a , b 两个点 . 限制完了之后 , 因为 n <= 50 , 我们可以直接从从小到大枚举 , 一个一个增广 . 具体建图 : boy_a( i ) -> boy_b( i ) ( cap : k ) , girl_b( i ) -> girl_a( i

bzoj 1305: [CQOI2009]dance跳舞

题目链接 bzoj 1305: [CQOI2009]dance跳舞 题解 男,女生拆点A1A2,B1B2,拆成两点间分别连容量为K的边,限制与不喜欢的人跳舞的数量 A1连接源点容量为x,B1连接汇点容量为x,x即为歌曲数目 对与相互喜欢的男女直在A1,B1间接连容量为1的边 对于相互不喜欢A2练到B2,容量为1 二分x的大小,每次重新构图,判断该图是否满流,慢流则说明该曲目可行 代码 #include<queue> #include<cstdio> #include<cstr

BZOJ 1305 [CQOI2009]dance跳舞(二分+网络流)

[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1305 [题目大意] 一次舞会有n个男孩和n个女孩. 每首曲子开始时,所有男孩和女孩恰好配成n对跳交谊舞. 每个男孩都不会和同一个女孩跳两首(或更多)舞曲. 有一些男孩女孩相互喜欢,而其他相互不喜欢(不会“单向喜欢”). 每个男孩最多只愿意和k个不喜欢的女孩跳舞, 而每个女孩也最多只愿意和k个不喜欢的男孩跳舞.给出每对男孩女孩是否相互喜欢的信息, 舞会最多能有几首舞曲? [题解] 我们

BZOJ 1305 CQOI2009 dance跳舞 二分答案+最大流

题目大意:给定n个男生和n个女生,一些互相喜欢而一些不.举行几次舞会,每次舞会要配成n对.不能有同样的组合出现.每一个人仅仅能与不喜欢的人跳k次舞,求最多举行几次舞会 将一个人拆成两个点.点1向点2连一条流量为k的边.两个人若互相喜欢则点1之间连边,不喜欢则点2之间连边 对于每个要验证的x值 将每个人的点1向源或汇连一条流量为x的边 然后二分答案跑最大流就可以 #include<cstdio> #include<cstring> #include<iostream> #

【BZOJ 1305】 [CQOI2009]dance跳舞

1305: [CQOI2009]dance跳舞 Time Limit: 5 Sec  Memory Limit: 162 MB Submit: 1586  Solved: 666 [Submit][Status] Description 一次舞会有n个男孩和n个女孩.每首曲子开始时,所有男孩和女孩恰好配成n对跳交谊舞.每个男孩都不会和同一个女孩跳两首(或更多)舞曲.有一些男孩女孩相互喜欢,而其他相互不喜欢(不会"单向喜欢").每个男孩最多只愿意和k个不喜欢的女孩跳舞,而每个女孩也最多只

BZOJ-1305: [CQOI2009]dance跳舞(最大流 增加超级源汇+分裂点+二分 水题)

1305: [CQOI2009]dance跳舞 Time Limit: 5 Sec  Memory Limit: 162 MB Description 一次舞会有n个男孩和n个女孩.每首曲子开始时,所有男孩和女孩恰好配成n对跳交谊舞.每个男孩都不会和同一个女孩跳两首(或更多)舞曲.有一些男孩女孩相互喜欢,而其他相互不喜欢(不会“单向喜欢”).每个男孩最多只愿意和k个不喜欢的女孩跳舞,而每个女孩也最多只愿意和k个不喜欢的男孩跳舞.给出每对男孩女孩是否相互喜欢的信息,舞会最多能有几首舞曲? Inpu

bzoj 1305: [CQOI2009]dance 二分+網絡流判定

1305: [CQOI2009]dance跳舞 Time Limit: 5 Sec  Memory Limit: 162 MBSubmit: 1340  Solved: 581[Submit][Status] Description 一次舞会有n个男孩和n个女孩.每首曲子开始时,所有男孩和女孩恰好配成n对跳交谊舞.每个男孩都不会和同一个女孩跳两首(或更多)舞曲.有一些男孩女孩相互喜欢,而其他相互不喜欢(不会“单向喜欢”).每个男孩最多只愿意和k个不喜欢的女孩跳舞,而每个女孩也最多只愿意和k个不喜

bzoj千题计划130:bzoj1305: [CQOI2009]dance跳舞

http://www.lydsy.com/JudgeOnline/problem.php?id=1305 每个人拆为喜欢(yes)和不喜欢(no)两个点 二分答案 1.每两个人之间只能跳一次 喜欢则 男yes i 向 女yes j 连流量为1的边 不喜欢则 男no i 向 女no j 连流量为1的边 2.最多与k个不喜欢的人跳 男yes i 向 男no i 连流量为 k的边 女no j 向 女yes j 连流量为k 的边 3. 源点向 男yes i 连流量mid 的边 女yes j 向汇点连流量

bzoj1305: [CQOI2009]dance跳舞(二分答案+网络流)

1305: [CQOI2009]dance跳舞 题目:传送门 题解: 一眼网络流基础建模...然后就GG了 二分答案+拆点建边+最大流判断: 把男女生拆为男1,男2,女1,女2 1.男1和男2还有女1和女2之间连边,流量为约束条件k 2.st连男1,女2连ed,流量为二分的mid 3.如果男生i喜欢女生j,就将男1与女2相连(不在约束条件内) 4.如果不喜欢,就将男2与女1相连(在约束条件内) 代码: 1 #include<cstdio> 2 #include<cstring> 3