题意

Frank是一个思想有些保守的高中老师。有一次,他需要带一些学生出去旅行,但又怕其中一些学生在旅行中萌生爱意。为了降低这种事情发生的概率,他决定确保带出去的任意两个学生至少要满足下面四条中的一条。

1.身高相差大于40厘米

2.性别相同

3.最喜欢的音乐属于不同类型

4.最喜欢的体育比赛相同

你的任务是帮助Frank挑选尽量多的学生,使得任意两个学生至少满足上述条件中的一条。

分析

这个模型叫二分图的最大独立集。既选择尽量多的结点,使得任意两个结点不相邻(既任意一条边的两个端点不会被同时选中)。最大独立集与最小覆盖是互补的,因此答案就是结点总数减去最大匹配数。

建模:将每个人看作一个结点,如果两个人四个条件都不满足,就意味着他们不能同时被选择,连一条无向边。这样,问题就转换为求这个图的最大独立集。因为他们每个人不是男生就是女生,所以这个图是二分图。

按照惯例,我依然是用网络流来做的最大匹配。原因依然是不会KM···等哪天(8012年)我学会来并且心情好可能会来补一下KM的代码。

  1 #include <cstdio>
  2 #include <algorithm>
  3 #include <cstring>
  4 #include <iostream>
  5 #include <cmath>
  6 #include <queue>
  7
  8 using namespace std;
  9 const int maxn=1000+10;
 10 const int maxm=2000000+100;
 11
 12 const int INF=2147000000;
 13 struct Dinic{
 14     int head[maxn],Next[maxm],to[maxm],cap[maxm],flow[maxm];
 15     int sz,n,m,s,t;
 16     bool vis[maxn];
 17     int cur[maxn],d[maxn];
 18     void init(int n){
 19         this->n=n;
 20         memset(head,-1,sizeof(head));
 21         this->sz=-1;
 22     }
 23     void add_edge(int a,int b,int c){
 24         ++sz;
 25         to[sz]=b;
 26         cap[sz]=c;flow[sz]=0;
 27         Next[sz]=head[a];head[a]=sz;
 28         ++sz;
 29         to[sz]=a;
 30         cap[sz]=c;flow[sz]=c;
 31         Next[sz]=head[b];head[b]=sz;
 32     }
 33     bool BFS(){
 34         memset(vis,0,sizeof(vis));
 35         queue<int>Q;
 36         vis[s]=1;
 37         d[s]=0;
 38         Q.push(s);
 39         while(!Q.empty()){
 40             int u=Q.front();Q.pop();
 41             for(int i=head[u];i!=-1;i=Next[i]){
 42                 int v=to[i];
 43                 if(!vis[v]&&cap[i]>flow[i]){
 44                     vis[v]=1;
 45                     d[v]=d[u]+1;
 46                     Q.push(v);
 47                 }
 48             }
 49         }
 50         return vis[t];
 51    }
 52     int DFS(int x,int a){
 53         if(x==t||a==0)return a;
 54         int Flow=0,f;
 55         for(int& i=cur[x];i!=-1;i=Next[i]){
 56             int v=to[i];
 57             if(d[v]==d[x]+1&&(f=DFS(v,min(a,cap[i]-flow[i])))>0){
 58                 Flow+=f;
 59                 flow[i]+=f;
 60                 flow[i^1]-=f;
 61                 a-=f;
 62                 if(a==0)break;
 63             }
 64         }
 65         return Flow;
 66     }
 67     int Maxflow(int s,int t){
 68         this->s=s,this->t=t;
 69         int Flow=0;
 70         while(BFS()){
 71             for(int i=0;i<=n;i++)
 72              cur[i]=head[i];
 73
 74             Flow+=DFS(s,INF);
 75         }
 76         return Flow;
 77     }
 78 }dinic;
 79 int T,n;
 80 int high[maxn];
 81 char sex[maxn];
 82 string mus[maxn],phy[maxn];
 83 int main(){
 84     scanf("%d",&T);
 85     for(int t=1;t<=T;t++){
 86         scanf("%d",&n);
 87         for(int i=1;i<=n;i++){
 88             scanf("%d %c",&high[i],&sex[i]);
 89             cin>>mus[i]>>phy[i];
 90         }
 91         dinic.init(n+2);
 92         for(int i=1;i<=n;i++){
 93             if(sex[i]=='M'){
 94                 for(int j=1;j<=n;j++){
 95                     if(sex[j]=='F'&&abs(high[i]-high[j])<=40&&mus[i]==mus[j]&&phy[i]!=phy[j]){
 96                             dinic.add_edge(i,j,1);
 97                     }
 98                 }
 99             }
100         }
101         for(int i=1;i<=n;i++){
102             if(sex[i]=='M')
103                 dinic.add_edge(0,i,1);
104         }
105         for(int i=1;i<=n;i++){
106             if(sex[i]=='F')
107                 dinic.add_edge(i,n+1,1);
108         }
109         int ans=dinic.Maxflow(0,n+1);
110         printf("%d\n",n-ans);
111     }
112 return 0;
113 }

View Code

其实我觉得这道题不知道这个模型只是考虑最小割的话也能想。去除最少的学生使得剩下的所有学生之间任意两个之间都满足四个条件中的一条。我们在学生之间连的边是这两个学生间只能选一个或者都不选,那个割掉这条边就代表去掉这两个学生中的一个。所以可以求出最小割来以后,用总的人数n减掉最小割。

哇万能的最小割我感觉这样想比背什么模型好多了啊!

转载于:https://www.cnblogs.com/LQLlulu/p/9307386.html

【LA3415 训练指南】保守的老师 【二分图最大独立集,最小割】相关推荐

  1. 训练指南 UVALive - 4043(二分图匹配 + KM算法)

    layout: post title: 训练指南 UVALive - 4043(二分图匹配 + KM算法) author: "luowentaoaa" catalog: true ...

  2. 【网络流24题】I、 方格取数问题(二分图的最大独立集/最小割)

    I. 方格取数问题(二分图的最大独立集/最小割) [问题分析] 二分图点权最大独立集,转化为最小割模型,从而用最大流解决. [建模方法] 首先把棋盘黑白染色,使相邻格子颜色不同,所有黑色格子看做二分图 ...

  3. 734. [网络流24题] 方格取数问题 二分图点权最大独立集/最小割/最大流

    «问题描述: 在一个有m*n 个方格的棋盘中,每个方格中有一个正整数.现要从方格中取数,使任 意2 个数所在方格没有公共边,且取出的数的总和最大.试设计一个满足要求的取数算法. «编程任务: 对于给定 ...

  4. HDU1569 方格取数(2)(二分图带权最大独立集 - 最小割应用)

    题目链接 结论 ∣带权最大独立集∣=∣点权之和∣−∣最小割∣=∣点权之和∣−∣最大流∣|带权最大独立集| = |点权之和| - |最小割| = |点权之和| - |最大流|∣带权最大独立集∣=∣点权之 ...

  5. 刘汝佳训练指南《网络流》专题 BY 9974

    最近一直在做白书上的网络流, 做得几乎差不多了,在这小小总结一下. 已经忽略很水的题.差不多按难度排序了. UVa 11248                   增广小优化. UVa 1306(LA ...

  6. 训练指南——数学专题一的总结

    差不多一个星期过去了,在这一个多星期里,我做了一个数学专题和两场训练赛,要说对自己的感觉,只能说很差劲,开始的时候以为环境会比现在宽松很多,后来才发现想法是错误的,实验室室里室一种紧张的气氛,感觉就像 ...

  7. 《算法竞赛入门经典训练指南》pdf

    下载地址:网盘下载 基本介绍 编辑 内容简介 <算法竞赛入门经典:训练指南>题目多选自近年来ACM/ICPC区域赛和总决赛真题,内容全面,信息量大,覆盖了常见算法竞赛中的大多数细分知识点. ...

  8. 一根绳子从一头烧需30时分钟_小学生一分钟跳绳满分训练指南

    小学生一分钟跳绳测试项目是体育老师和众多家庭的痛,大多数的孩子1分钟跳不到60个.而跳绳又是体质健康测试里占分比较高的一个项目,所以很多宝妈/宝爸们很是焦虑.因此,我们根据平时儿童跳绳训练的经验和网上 ...

  9. 算法竞赛入门经典——训练指南

    <算法竞赛入门经典--训练指南> 基本信息 作者: 刘汝佳 陈锋 [作译者介绍] 丛书名: 算法艺术与信息学竞赛 出版社:清华大学出版社 ISBN:9787302291077 上架时间:2 ...

最新文章

  1. mysql外键写了会怎么样_mysql使用外键会影响性能吗
  2. 高校促进“智慧城市”信息化建设策略研究
  3. linux下查找命令汇总(转)
  4. 算个欧拉函数给大家助助兴(米勒拉宾(判断素数)+Pollard_rho(求一个大数的因子 ))
  5. echarts --- 多折线图按段显示颜色规则订制
  6. android webview js 交互框架,自定义android混合框架开发实践1:实现基础andorid和webview交互...
  7. 卸载程序_App Cleaner Pro for Mac v6.10.1 程序卸载 直装版
  8. 为什么200M宽带还是会很慢?
  9. JavaEE——Mybatis(5)--resultMap自定义结果集封装
  10. Atitit.ide代码块折叠插件 eclipse
  11. Kaggle注册无法进行人机验证You did not enter the correct captcha
  12. java快速生成接口文档方法总结
  13. 【钉钉-场景化能力包】阿里商旅助力费控报销
  14. MATLAB计算13195的约数,最大约数算法 | Delphi论坛 | Delphi Forum - We Delphi
  15. 皮尔逊相关系数,斯皮尔曼等级相关系数,(易错!!)假设检验 ,SPSS
  16. “分集”与“复用”辨析
  17. Web Directions South 2012的四个大创意
  18. 单例模式singleton
  19. 推荐一款特别厉害的在线工具,程序员的百宝箱
  20. 开源数据可视化 datart-自定义Jquery图表插件教程

热门文章

  1. 华为双系统是鸿蒙系统吗,华为p50pro是鸿蒙系统吗-华为p50pro有双系统吗
  2. 【BZOJ - 3224】普通平衡树(Splay模板题)
  3. 【牛客 - 551G】CSL的训练计划(二分 + 拓扑排序 + 优化卡常)
  4. 【CodeForces - 523C】Name Quest (模拟)
  5. linux 源码安装mysql5.7_linux安装mysql5.7.27
  6. IO流配置文件,键值对Properties 的读取
  7. leetcode445. 两数相加 II
  8. leetcode914. 卡牌分组
  9. leetcode586. 订单最多的客户(SQL)
  10. C++中volatile关键字