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

题意

  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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/451988.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

行车记录仪稳定方案:TC358778XBG:RGB转MIPI DSI芯片,M-Star标配IC

原厂&#xff1a;Toshiba型号&#xff1a;TC358778XBG功能&#xff1a;TC358778XBG是一颗将RGB信号转换成MIPI DSI的芯片&#xff0c;最高分辨率支持到1920x1200&#xff0c;其应用图如下&#xff1a;产品特征&#xff1a;MIPI接口&#xff1a;&#xff08;1&#xff09;、支持…

java.sql.SQLException: 无法转换为内部表示之解决

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到教程。 这个错是因为 数据库中字段类型和程序中该字段类型不一致。 比如程序将某字段当做Integer类型&#xff0c; 而数据库存储又使用另外一…

网络爬虫--17.【BeautifuSoup4实战】爬取腾讯社招

文章目录一.要求二.代码示例一.要求 以腾讯社招页面来做演示&#xff1a;http://hr.tencent.com/position.php?&start10#a 使用BeautifuSoup4解析器&#xff0c;将招聘网页上的职位名称、职位类别、招聘人数、工作地点、发布时间&#xff0c;以及每个职位详情的点击链接…

彻底搞清楚Android中的 Attr

版权声明&#xff1a;本文为sydMobile原创文章&#xff0c;转载请务必注明出处&#xff01; https://blog.csdn.net/sydMobile/article/details/79978187 相信这个词对于Android开发者来说十分熟悉了&#xff0c;那么你对他到底有多了解呢&#xff1f; 回忆起我刚开始接触Andr…

D. Relatively Prime Graph

Lets call an undirected graph G(V,E)G(V,E) relatively prime if and only if for each edge (v,u)∈E(v,u)∈E GCD(v,u)1GCD(v,u)1 (the greatest common divisor of vv and uu is 11). If there is no edge between some pair of vertices vv and uu then the value of GC…

网络爬虫--19.【Scrapy-Redis实战】分布式爬虫爬取房天下--环境准备

文章目录0. 思路一. 虚拟机Ubuntu0中安装Redis二. 虚拟机Ubuntu1中安装Redis三. Windows服务器上安装Redis四. 安装cmder五. 安装RedisDesktopManager六. 修改Windows中的配置文件redis.windows.conf七. Ubuntu连接Windows上 的Redis服务器-----------------------------------…

tkinter中scale拖拉改变值控件(十一)

scale拖拉改变值控件 使用户通过拖拽改变值 简单的实现&#xff1a; 1 import tkinter2 3 wuya tkinter.Tk() 4 wuya.title("wuya") 5 wuya.geometry("300x2001020") 6 7 8 # 创建对象 9 scale1 tkinter.Scale(wuya, from_0, to100) 10 scale1.pac…

计算机图形学理论(4):缓冲区

本系列根据国外一个图形小哥的讲解为本&#xff0c;整合互联网的一些资料&#xff0c;结合自己的一些理解。 什么是缓冲区&#xff1f; 缓冲区是保存某些数据的临时存储空间。 为什么我们需要缓冲区&#xff1f;原因很简单&#xff0c;当数据量很大时&#xff0c;因为计算机无…

网络爬虫--20.【Scrapy-Redis实战】分布式爬虫获取房天下--代码实现

文章目录一. 案例介绍二.创建项目三. settings.py配置四. 详细代码五. 部署1. windows环境下生成requirements.txt文件2. xshell连接ubuntu服务器并安装依赖环境3. 修改部分代码4. 上传代码至服务器并运行一. 案例介绍 爬取房天下&#xff08;https://www1.fang.com/&#xff…

同一台电脑安装python2python3

【安装之前&#xff0c;先了解一下概念】 python是什么&#xff1f; Python是一种面向对象的解释型计算机程序设计语言&#xff0c;由荷兰人Guido van Rossum于1989年发明&#xff0c;第一个公开发行版发行于1991年。 Python是纯粹的自由软件&#xff0c; 源代码和解释器CPytho…

程序员的常见健康问题

其实这些问题不仅见于程序员&#xff0c;其他长期经常坐在电脑前的职场人士&#xff08;比如&#xff1a;网络编辑、站长等&#xff09;&#xff0c;都会有其中的某些健康问题。希望从事这些行业的朋友&#xff0c;对自己的健康问题&#xff0c;予以重视。以下是全文。 我最近…

网络爬虫--21.Scrapy知识点总结

文章目录一. Scrapy简介二. Scrapy架构图三. Scrapy框架模块功能四. 安装和文档五. 创建项目六. 创建爬虫一. Scrapy简介 二. Scrapy架构图 三. Scrapy框架模块功能 四. 安装和文档 中文文档&#xff1a;https://scrapy-chs.readthedocs.io/zh_CN/latest/intro/tutorial.html …

Ubuntu将在明年推出平板及手机系统

4月26日下午消息&#xff0c;知名Linux厂商Canonical今天正式发布Ubuntu 12.04版开源操作系统。Ubuntu中国首席代表于立强透露&#xff0c;针对平板电脑的Ubuntu操作系统将在明年推出。 Ubuntu 12.04版开源操作系统发布 Ubuntu操作系统是一款开源操作系统&#xff0c;主要与OE…

Android Studio 超级简单的打包生成apk

为什么要打包&#xff1a; apk文件就是一个包&#xff0c;打包就是要生成apk文件&#xff0c;有了apk别人才能安装使用。打包分debug版和release包&#xff0c;通常所说的打包指生成release版的apk&#xff0c;release版的apk会比debug版的小&#xff0c;release版的还会进行混…

推荐16款最棒的Visual Studio插件

Visual Studio是微软公司推出的开发环境&#xff0c;Visual Studio可以用来创建Windows平台下的Windows应用程序和网络应用程序&#xff0c;也可以用来创建网络服务、智能设备应用程序和Office插件。 本文介绍16款最棒的Visual Studio扩展&#xff1a; 1. DevColor Extension…

网络爬虫--22.【CrawlSpider实战】实现微信小程序社区爬虫

文章目录一. CrawlSpider二. CrawlSpider案例1. 目录结构2. wxapp_spider.py3. items.py4. pipelines.py5. settings.py6. start.py三. 重点总结一. CrawlSpider 现实情况下&#xff0c;我们需要对满足某个特定条件的url进行爬取&#xff0c;这时候就可以通过CrawlSpider完成。…

怎么安装Scrapy框架以及安装时出现的一系列错误(win7 64位 python3 pycharm)

因为要学习爬虫&#xff0c;就打算安装Scrapy框架&#xff0c;以下是我安装该模块的步骤&#xff0c;适合于刚入门的小白&#xff1a; 一、打开pycharm&#xff0c;依次点击File---->setting---->Project----->Project Interpreter&#xff0c;打开后&#xff0c;可以…

xpath-helper: 谷歌浏览器安装xpath helper 插件

1.下载文件xpath-helper.crx xpath链接&#xff1a;https://pan.baidu.com/s/1dFgzBSd 密码&#xff1a;zwvb&#xff0c;感谢这位网友&#xff0c;我从这拿到了 2.在Google浏览器里边找到这个“扩展程序”选项菜单即可。 3.然后就会进入到扩展插件的界面了,把下载好的离线插件…

网络爬虫--23.动态网页数据抓取

文章目录一. Ajax二. 获取Ajax数据的方式三. seleniumchromedriver获取动态数据四. selenium基本操作一. Ajax 二. 获取Ajax数据的方式 三. seleniumchromedriver获取动态数据 selenium文档&#xff1a;https://selenium-python.readthedocs.io/installation.html 四. sele…