有n个人, 其中有男生和女生,接着有n行,分别给出了每一个人暗恋的对象(不止暗恋一个)

现在要从这n个人中找出一个最大集合,满足这个集合中的任意2个人,都没有暗恋这种关系。

输出集合的元素个数。

刚开始想,把人看成顶点,若有暗恋的关系,就连一条边,构成一个图

独立集的概念:一个图中两两互不相连的顶点集合

所以这道题,就是要求最大独立集

有:最大独立集+最小顶点覆盖=|V|(顶点的总个数)

那就求最小顶点覆盖了

根据题意:

暗恋的对象性别不同,所以a暗恋b,b暗恋c,c暗恋a这种关系不可能存在

也就是说,这个图的顶点可以根据性别分成2个集合,男生和女生

即这是一个二分图

我们知道,在二分图中,最小顶点覆盖=最大匹配

则:最大独立集=|V|-最大匹配

所以思路就清晰了:

1.黑白染色:确定男生和女生

2.建图:s连边到所有男生,所有女生连边到t,若男生i和女生j有关系,则连一条边,边的容量都是1

3.二分匹配转化为最大流求解

4.|V|-最大匹配

 #include<cstdio>
#include<cstring>
#include<vector>
#include<queue> using namespace std; const int maxn=;
const int inf=0x3f3f3f3f;
int s;
int t; inline int min(int x,int y)
{
return x<y?x:y;
} struct Edge
{
int to,cap,rev;
};
vector<Edge>edge[maxn];
int level[maxn];
int iter[maxn];
int dye[maxn]; void addedge(int from,int to,int cap)
{
edge[from].push_back((Edge){to,cap,edge[to].size()});
edge[to].push_back((Edge){from,,edge[from].size()-});
} struct Edge1
{
int to,next;
};
Edge1 edge1[maxn*];
int head[maxn],tot; void init()
{
memset(head,-,sizeof head);
tot=;
memset(dye,-,sizeof dye);
} void addedge1(int from,int to)
{
edge1[tot].to=to;
edge1[tot].next=head[from];
head[from]=tot++;
} void get_dye(int u,int pre)
{
if(pre==-)
dye[u]=;
else
dye[u]=!dye[pre];
for(int i=head[u];~i;i=edge1[i].next)
{
int v=edge1[i].to;
if(v==pre)
continue;
if(dye[v]!=-)
continue;
get_dye(v,u);
}
} void build_graph(int n)
{
s=n;
t=n+;
for(int i=;i<=t;i++)
edge[i].clear();
for(int i=;i<n;i++)
{
if(dye[i]==)
{
addedge(s,i,);
for(int j=head[i];~j;j=edge1[j].next)
{
int v=edge1[j].to;
addedge(i,v,);
}
}
else
addedge(i,t,);
}
} void bfs()
{
memset(level,-,sizeof level);
queue<int>que;
while(!que.empty())
que.pop();
que.push(s);
level[s]=;
while(!que.empty())
{
int u=que.front();
que.pop();
for(int i=;i<edge[u].size();i++)
{
Edge &e=edge[u][i];
if(e.cap>&&level[e.to]<)
{
level[e.to]=level[u]+;
que.push(e.to);
}
}
}
} int dfs(int u,int f)
{
if(u==t)
return f;
for(int &i=iter[u];i<edge[u].size();i++)
{
Edge &e=edge[u][i];
if(e.cap>&&level[e.to]>level[u])
{
int d=dfs(e.to,min(e.cap,f));
if(d)
{
e.cap-=d;
edge[e.to][e.rev].cap+=d;
return d;
}
}
}
return ;
} int solve()
{
int flow=;
while()
{
bfs();
if(level[t]<)
return flow;
memset(iter,,sizeof iter);
int f;
while(f=dfs(s,inf))
{
flow+=f;
}
}
} int main()
{
int n;
while(~scanf("%d",&n))
{
init();
for(int i=;i<n;i++)
{
int j;
char ch;
scanf("%d%c",&j,&ch);
char str[];
scanf("%s",str);
int len=strlen(str);
int num=;
for(int k=;k<len-;k++)
{
num=num*+(str[k]-'');
}
for(int k=;k<num;k++)
{
int tmp;
scanf("%d",&tmp);
addedge1(j,tmp);
addedge1(tmp,j);
}
} for(int i=;i<n;i++)
{
if(dye[i]==-)
get_dye(i,-);
}
build_graph(n);
printf("%d\n",n-solve());
}
return ;
}

POJ 1466 Girls and Boys 黑白染色 + 二分匹配 (最大独立集) 好题的更多相关文章

  1. poj 1466 Girls and Boys(二分图的最大独立集)

    http://poj.org/problem?id=1466 Girls and Boys Time Limit: 5000MS   Memory Limit: 10000K Total Submis ...

  2. POJ 1466 Girls and Boys

    Girls and Boys Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://poj.org/problem?id=1466 Descripti ...

  3. poj 1466 Girls and Boys 二分图的最大匹配

    Girls and Boys Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://poj.org/problem?id=1466 Descripti ...

  4. POJ 1466 Girls and Boys (匈牙利算法 最大独立集)

    Girls and Boys Time Limit: 5000MS   Memory Limit: 10000K Total Submissions: 10912   Accepted: 4887 D ...

  5. 网络流(最大独立点集):POJ 1466 Girls and Boys

    Girls and Boys Time Limit: 5000ms Memory Limit: 10000KB This problem will be judged on PKU. Original ...

  6. poj 1466 Girls and Boys (最大独立集)

    链接:poj 1466 题意:有n个学生,每一个学生都和一些人有关系,如今要你找出最大的人数.使得这些人之间没关系 思路:求最大独立集,最大独立集=点数-最大匹配数 分析:建图时应该是一边是男生的点, ...

  7. poj 1466 Girls and Boys(二分匹配之最大独立集)

    Description In the second year of the university somebody started a study on the romantic relations ...

  8. POJ 1466 Girls and Boys(二分图匹配)

    [题目链接] http://poj.org/problem?id=1466 [题目大意] 给出一些人和他们所喜欢的人,两个人相互喜欢就能配成一对, 问最后没有配对的人的最少数量 [题解] 求最少数量, ...

  9. POJ 1466 Girls and Boys (ZOJ 1137 )最大独立点集

    http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=137 http://poj.org/problem?id=1466 题目大意: ...

随机推荐

  1. 3-4 rpm包查询

    概述:yum不能查询已经安装好的rpm包, 就算采用了yum来进行安装,查询方法还是依赖rpm包的查询, 因此rpm包的查询十分常用和重要 1.查询是否安装 <1>rpm -q 包名(不是 ...

  2. 顺序表及其多种实现方式 --- C/C++

    所谓顺序表,即线性表的顺序存储结构.下面给出的是数据结构---线性表的定义. ADT List{ 数据对象: 线性表的数据对象的集合为{a1,a2,a3,...,an},每个元素的类型为ElemTyp ...

  3. POJ 3041 Asteroids(最小点覆盖集)

                                                                      Asteroids Time Limit: 1000MS   Mem ...

  4. c程序代码优化的一些方法

    我认为一个好的用于科学计算的程序代码应该:算法漂亮精妙,程序简洁易懂,运算快速,节省内存.这里有的地方是矛盾的,比如简洁vs易懂,时间vs空间,找个平衡吧.目前来看时间要比空间宝贵一些.写程序分几步: ...

  5. Oracle Enterprise Manager Cloud Control 12c的安装

    这里,我介绍的是Cloud Control 12c的安装.之前的版本叫Grid Control,这也是11g里面的叫法.在我写这篇博客时,Cloud Control 13c已经出来了,这个版本后续我再 ...

  6. MySQLdb

    MySQLdb的使用 #mysql> create table `account`( # -> `acctid` int(11) default null comment 'XXXX', ...

  7. CorelDRAW 实现蒙版效果的方法

    CorelDRAW能够实现很多意想不到的小效果,其中包括了位图图像软件的处理功能,蒙版效果就是其中的一项.作为矢量图形处理软件,从理论上讲它并不具备蒙板技术,然而只是我们平常没有用到而已,利用图框精确 ...

  8. jquery 使用attr() 函数对复选框无效的原因,javascript那些事儿——properties和attributes

    复选框是网站开发的时候经常用到的网页标签之一,常见的在页面上对复选框的操作包括取值和修改复选框的状态.在jquery中,常见的操作标签的值得函数为attr,然而在操作复选框的时候,通常采用的却是pro ...

  9. 解决Win10默认占用80端口

    方案1: 以管理员身份运行cmd;输入net stop http;如果提示是否真的需要停止这些服务,则选择“Y”;完成后输入:sc config http start=disabled 方案2: Ct ...

  10. Java collection 的一些介绍 集合

    collections主要提供一些,排序的算法,随机的,反向的,  collection 是容器的祖先接口 线性表,链表,哈希表是常用的数据结构,在进行Java开发时,JDK已经为我们提供了一系列相应 ...