【题目链接】 http://www.lydsy.com/JudgeOnline/problem.php?id=1305

【题目大意】

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

【题解】

  我们发现答案具有单调性,因此可以二分检验。
  由于每个男孩选择的女孩以及每个女孩选择男孩分为两个部分,选择喜欢或者不喜欢的
  因此我们将其拆为两个部分,对于不喜欢的部分有k流限制,
  记喜欢为X集合,不喜欢为Y集合,我们对于喜欢的部分男孩的X和女孩的X之间连流量为1的边
  对于不喜欢的部分,男孩的Y和女孩的Y之间连流量为1的边
  男孩的X往Y连流量为K的边表示不喜欢的流量限制,女孩的Y往X连K的边表示接受的不喜欢的流量限制
  对于二分值a,我们从源点引a的流量到每个男孩的X点,从女孩的X点引a的流量到汇点,
  那么满流即表示答案可行。

【代码】

  1. #include <cstdio>
  2. #include <cstring>
  3. #include <vector>
  4. #include <queue>
  5. using namespace std;
  6. const int INF=0x3f3f3f3f;
  7. const int MAX_V=20010;
  8. struct edge{int to,cap,rev;};
  9. vector<edge> G[MAX_V];
  10. int level[MAX_V],iter[MAX_V];
  11. void add_edge(int from,int to,int cap){
  12. G[from].push_back((edge){to,cap,G[to].size()});
  13. G[to].push_back((edge){from,0,G[from].size()-1});
  14. }
  15. void bfs(int s){
  16. memset(level,-1,sizeof(level));
  17. queue<int> que;
  18. level[s]=0;
  19. que.push(s);
  20. while(!que.empty()){
  21. int v=que.front(); que.pop();
  22. for(int i=0;i<G[v].size();i++){
  23. edge &e=G[v][i];
  24. if(e.cap>0&&level[e.to]<0){
  25. level[e.to]=level[v]+1;
  26. que.push(e.to);
  27. }
  28. }
  29. }
  30. }
  31. int dfs(int v,int t,int f){
  32. if(v==t)return f;
  33. for(int &i=iter[v];i<G[v].size();i++){
  34. edge &e=G[v][i];
  35. if(e.cap>0&&level[v]<level[e.to]){
  36. int d=dfs(e.to,t,min(f,e.cap));
  37. if(d>0){
  38. e.cap-=d;
  39. G[e.to][e.rev].cap+=d;
  40. return d;
  41. }
  42. }
  43. }return 0;
  44. }
  45. int max_flow(int s,int t){
  46. int flow=0;
  47. for(;;){
  48. bfs(s);
  49. if(level[t]<0)return flow;
  50. memset(iter,0,sizeof(iter));
  51. int f;
  52. while((f=dfs(s,t,INF))>0){
  53. flow+=f;
  54. }
  55. }
  56. }
  57. int N,K;
  58. char mp[55][55];
  59. bool check(int x){
  60. int s=4*N,t=s+1;
  61. for(int i=0;i<=t;i++)G[i].clear();
  62. for(int i=0;i<N;i++){
  63. for(int j=0;j<N;j++){
  64. if(mp[i][j]=='Y')add_edge(i,j+N*2,1);
  65. else add_edge(i+N,j+N*3,1);
  66. }add_edge(i,i+N,K);
  67. add_edge(i+N*3,i+N*2,K);
  68. add_edge(s,i,x);
  69. add_edge(i+N*2,t,x);
  70. }return max_flow(s,t)==n*x;
  71. }
  72. int main(){
  73. while(~scanf("%d%d",&N,&K)){
  74. for(int i=0;i<N;i++)scanf("%s",mp[i]);
  75. int l=1,r=N,ans=0;
  76. while(l<=r){
  77. int mid=(l+r)>>1;
  78. if(check(mid))ans=mid,r=mid-1;
  79. else l=mid+1;
  80. }printf("%d\n",ans);
  81. }return 0;
  82. }

BZOJ 1305 [CQOI2009]dance跳舞(二分+网络流)的更多相关文章

  1. BZOJ 1305: [CQOI2009]dance跳舞 二分+最大流

    1305: [CQOI2009]dance跳舞 Description 一次舞会有n个男孩和n个女孩.每首曲子开始时,所有男孩和女孩恰好配成n对跳交谊舞.每个男孩都不会和同一个女孩跳两首(或更多)舞曲 ...

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

    题目大意:给定n个男生和n个女生,一些互相喜欢而一些不.举行几次舞会,每次舞会要配成n对.不能有同样的组合出现.每一个人仅仅能与不喜欢的人跳k次舞,求最多举行几次舞会 将一个人拆成两个点.点1向点2连 ...

  3. bzoj 1305: [CQOI2009]dance跳舞

    题目链接 bzoj 1305: [CQOI2009]dance跳舞 题解 男,女生拆点A1A2,B1B2,拆成两点间分别连容量为K的边,限制与不喜欢的人跳舞的数量 A1连接源点容量为x,B1连接汇点容 ...

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

    云神代码很短...0 ms过的...看了代码 , 大概是贪心... orz 我不会证 数据这么小乱搞就可以了吧... ←_← 这道题网络流还是可以写的... 既然限制了最多只能和 k 个不喜欢的人da ...

  5. BZOJ 1305: [CQOI2009]dance跳舞 网络最大流_二分答案_建模

    Description 一次舞会有n个男孩和n个女孩.每首曲子开始时,所有男孩和女孩恰好配成n对跳交谊舞.每个男孩都不会和同一个女孩跳两首(或更多)舞曲.有一些男孩女孩相互喜欢,而其他相互不喜欢(不会 ...

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

    1305: [CQOI2009]dance跳舞 Time Limit: 5 Sec  Memory Limit: 162 MBSubmit: 1340  Solved: 581[Submit][Sta ...

  7. BZOJ1305 [CQOI2009]dance跳舞 【网络流】

    1305: [CQOI2009]dance跳舞 Time Limit: 5 Sec  Memory Limit: 162 MB Submit: 3714  Solved: 1572 [Submit][ ...

  8. 1305: [CQOI2009]dance跳舞 - BZOJ

    Description 一次舞会有n个男孩和n个女孩.每首曲子开始时,所有男孩和女孩恰好配成n对跳交谊舞.每个男孩都不会和同一个女孩跳两首(或更多)舞曲.有一些男孩女孩相互喜欢,而其他相互不喜欢(不会 ...

  9. BZOJ 1305:dance跳舞(二分+最大流)

    一次舞会有n个男孩和n个女孩.每首曲子开始时,所有男孩和女孩恰好配成n对跳交谊舞.每个男孩都不会和同一个女孩跳两首(或更多)舞曲.有一些男孩女孩相互喜欢,而其他相互不喜欢(不会“单向喜欢”).每个男孩 ...

随机推荐

  1. POj 2104 K-th Number (分桶法+线段树)

    题目链接 Description You are working for Macrohard company in data structures department. After failing ...

  2. mssql手工注入2

    --+ 先说一些函数的说明: substring(str,start,len) 截取字符串的作用,第一个参数为要截取的字符串,第二个参数为从哪里开始截取,第三个参数为截取的长度 ascii(char) ...

  3. css position的值

    值 描述 absolute 生成绝对定位的元素,相对于 static 定位以外的第一个父元素进行定位. 元素的位置通过 "left", "top", " ...

  4. Optimizing subroutine calls based on architecture level of called subroutine

    A technique is provided for generating stubs. A processing circuit receives a call to a called funct ...

  5. Linux 内核同步之自旋锁与信号量的异同【转】

    转自:http://blog.csdn.net/liuxd3000/article/details/8567070 Linux 设备驱动中必须解决的一个问题是多个进程对共享资源的并发访问,并发访问会导 ...

  6. 1003: FFF团的情侣活动--课程作业--找出N个数字中唯一出现奇数次的数

    1003: FFF团的情侣活动 Time Limit: 1 Sec  Memory Limit: 2 MB Description 圣诞节快到了,Water作为大FFF团团长,组织许多对情侣进行电影院 ...

  7. Python中使用dom模块生成XML文件示例

    在Python中解析XML文件也有Dom和Sax两种方式,这里先介绍如何是使用Dom解析XML,这一篇文章是Dom生成XML文件,下一篇文章再继续介绍Dom解析XML文件. 在生成XML文件中,我们主 ...

  8. Hierarchical Attention Based Semi-supervised Network Representation Learning

    Hierarchical Attention Based Semi-supervised Network Representation Learning 1. 任务 给定:节点信息网络 目标:为每个节 ...

  9. Mysql 数据库学习笔记01查询

    1.数据查询基本操作 * 正则表达式查询: 字段名 regexp '匹配方式', select * from user where username regexp '^名'    -- 查询 姓名 名 ...

  10. CF868F Yet Another Minimization Problem

    题目描述: 给定一个序列,要把它分成k个子序列.每个子序列的费用是其中相同元素的对数.求所有子序列的费用之和的最小值. 输入格式:第一行输入n(序列长度)和k(需分子序列段数).下一行有n个数,序列的 ...