hdu4685 最大匹配可能性
题意:
给你n个王子和m个公主,每个王子可以和自己喜欢的公主结婚,问你在不影响最大匹配的前提下每个王子都可以去哪些公主.
思路:
所有王子向他喜欢的公主连一条边,然后匹配一遍,之后再每个匹配的公主连一条指向娶她的王子一条边,然后对于那些光棍王子和单身公主们,其实他们可以和任意他们喜欢的人匹配,因为可以让自己幸福,然后拆散一对别人已经匹配好的,虽然不道德,但是仍然满足总的最大匹配数不变,所以对于每一个光棍男我们就给他虚拟出一个女朋友,所有人都喜欢这个女的,但是这个女的只喜欢并且和该光棍男匹配,女的也类似这样弄,最后每个人都有匹配的对象了,此时我们只要跑一遍强连通,同一个强连通分量里的男和女可以随意匹配..
#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<stack>
#include<map> #define N_node 5000
#define N_edge 600000
using namespace std; typedef struct
{
int to ,next;
}STAR; STAR E[N_edge] ,_E[N_edge] ,__E[N_edge];
map<int ,map<int ,int> >mk_map;
int list[N_node] ,_list[N_node] ,tot , _tot;
int Belong[N_node] ,cout;
int mk_dfs[N_node] ,mk_gx[N_node];
int mk_M[550] ,mk_G[550];
stack<int>st; void add(int a, int b)
{
E[++tot].to = b;
E[tot].next = list[a];
list[a] = tot; _E[++_tot].to = a;
_E[_tot].next = _list[b];
_list[b] = _tot;
} int DFS_XYL(int s)
{
for(int k = list[s] ;k ;k = E[k].next)
{
int to = E[k].to;
if(mk_dfs[to]) continue;
mk_dfs[to] = 1;
if(mk_gx[to] == -1 || DFS_XYL(mk_gx[to]))
{
mk_gx[to] = s;
return 1;
}
}
return 0;
} void DFS_1(int s)
{
mk_dfs[s] = 1;
for(int k = list[s] ;k ;k = E[k].next)
{
int to = E[k].to;
if(mk_dfs[to]) continue;
DFS_1(to);
}
st.push(s);
} void DFS_2(int s)
{
mk_dfs[s] = 1;
Belong[s] = cout;
for(int k = _list[s] ;k ;k = _E[k].next)
{
int to = _E[k].to;
if(mk_dfs[to]) continue;
DFS_2(to);
}
} void solve(int cas)
{
int n ,m ,i ,j;
int a ,b ,c;
scanf("%d %d" ,&n ,&m);
memset(list ,0 ,sizeof(list));
memset(_list ,0 ,sizeof(_list));
tot = _tot = 1;
mk_map.clear();
for(i = 1 ;i <= n ;i ++)
{
scanf("%d" ,&c);
while(c--)
{
scanf("%d" ,&a);
add(i ,a + n);
mk_map[i][a] = 1;
}
}
memset(mk_gx ,255 ,sizeof(mk_gx));
int sum = 0;
for(i = 1 ;i <= n ;i ++)
{
memset(mk_dfs ,0 ,sizeof(mk_dfs));
mk_M[i] = DFS_XYL(i);
sum += mk_M[i];
}
for(i = 1 ;i <= m ;i ++)
{
if(mk_gx[i + n] == -1) mk_G[i] = 0;
else
{
mk_G[i] = 1;
add(i + n ,mk_gx[i + n]);
}
}
int nowid = n + m;
for(i = 1 ;i <= n ;i ++)
{
if(mk_M[i]) continue;
nowid ++;
for(j = 1 ;j <= n ;j ++)
add(j ,nowid);
add(nowid ,i);
}
for(i = 1 ;i <= m ;i ++)
{
if(mk_G[i]) continue;
nowid ++;
for(j = 1 ;j <= m ;j ++)
add(nowid ,j + n);
add(i + n ,nowid);
}
memset(mk_dfs ,0 ,sizeof(mk_dfs));
while(!st.empty()) st.pop();
for(i = 1 ;i <= nowid ;i ++)
{
if(mk_dfs[i]) continue;
DFS_1(i);
} cout = 0;
memset(mk_dfs ,0 ,sizeof(mk_dfs));
while(!st.empty())
{
int to = st.top();
st.pop();
if(mk_dfs[to]) continue;
cout ++;
DFS_2(to);
}
printf("Case #%d:\n" ,cas);
int tmp[550];
for(i = 1 ;i <= n ;i ++)
{
int tt = 0;
for(int k = list[i] ;k ;k = E[k].next)
{
int to = E[k].to;
if(mk_map[i][to - n] && Belong[i] == Belong[to])
tmp[++tt] = to - n;
}
sort(tmp + 1 ,tmp + tt + 1);
printf("%d" ,tt);
for(int k = 1 ;k <= tt ;k ++)
printf(" %d" ,tmp[k]);
printf("\n");
}
} int main ()
{
int cas = 1 ,t;
scanf("%d" ,&t);
while(t--)
{
solve(cas++);
}
return 0;
}
hdu4685 最大匹配可能性的更多相关文章
- POJ1904 强联通(最大匹配可能性)
题意: 有n个王子,n个公主,然后给你每个王子喜欢的公主,最后问你在不影响最大匹配的前提下,每个王子可以匹配那些公主. 思路: 是hdu4685的减弱版,之前研究过hdu468 ...
- UVALive 5033 I'm Telling the Truth 二分图最大匹配(略有修改)
I - I'm Telling the Truth Time Limit:3000MS Memory Limit:0KB 64bit IO Format:%lld & %llu ...
- Python学习实践------正向最大匹配中文分词
正向最大匹配分词: 1.加载词典文件到集合中,取词典文件中最大长度词的length 2.每次先在句子中按最大长度分割,然后判断分割的词是否存在字典中,存在则记录此词,调整起始点. 3.不存在则按最大长 ...
- UOJ79 一般图最大匹配
题目描述 从前一个和谐的班级,所有人都是搞OI的.有 nn 个是男生,有 00 个是女生.男生编号分别为 1,-,n1,-,n. 现在老师想把他们分成若干个两人小组写动态仙人掌,一个人负责搬砖另一个人 ...
- hdu 1281 二分图最大匹配
对N个可以放棋子的点(X1,Y1),(x2,Y2)......(Xn,Yn);我们把它竖着排看看~(当然X1可以对多个点~) X1 Y1 X2 Y2 X3 Y3 ..... Xn Yn ...
- POJ 2226二分图最大匹配
匈牙利算法是由匈牙利数学家Edmonds于1965年提出,因而得名.匈牙利算法是基于Hall定理中充分性证明的思想,它是二部图匹配最常见的算法,该算法的核心就是寻找增广路径,它是一种用增广路径求二分图 ...
- hdu-1179-二分图最大匹配
Ollivanders: Makers of Fine Wands since 382 BC. Time Limit: 2000/1000 MS (Java/Others) Memory Lim ...
- codevs1022 覆盖[Hungary 二分图最大匹配]
codevs1022 覆盖 有一个N×M的单位方格中,其中有些方格是水塘,其他方格是陆地.如果要用1×2的矩阵区覆盖(覆盖过程不容许有任何部分重叠)这个陆地,那么最多可以覆盖多少陆地面积. 输入描述 ...
- POJ1274 The Perfect Stall[二分图最大匹配]
The Perfect Stall Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 23911 Accepted: 106 ...
随机推荐
- MATLAB中FFT_HDL_Optimized模块定点(IEEE754单精度float格式)二进制与十进制转换实现
早些时间段,做了Matlab中FFT_HDL_Optimzed模块FFT HDL代码仿真,并与Xilinx Vivado自带的xfft IP进行单精度浮点比较(后面随笔叙述).因为FFT_HDL_Op ...
- 九. SpringCloud Stream消息驱动
1. 消息驱动概述 1.1 是什么 在实际应用中有很多消息中间件,比如现在企业里常用的有ActiveMQ.RabbitMQ.RocketMQ.Kafka等,学习所有这些消息中间件无疑需要大量时间经历成 ...
- C语言中指针和多维数组
指针和多维数组 数组名是特殊的指针 数组是一个特殊的指针,多维数组也是更为复杂的数组,它们的关系是什么样的呢? 我们通过一个简单的例子来比较形象的了解指针和多维数组: int a[2][3]; 这是一 ...
- 【pytest官方文档】解读fixtures - 7. Teardown处理,yield和addfinalizer
当我们运行测试函数时,我们希望确保测试函数在运行结束后,可以自己清理掉对环境的影响. 这样的话,它们就不会干扰任何其他的测试函数,更不会日积月累的留下越来越多的测试数据. 用过unittest的朋友相 ...
- FreeBSD 家图谱
https://cgit.freebsd.org/src/tree/share/misc/bsd-family-tree
- Android - Handler原理
Handler的主要作用是收发消息和切线程 功能一:收发消息 简单流程介绍 希望你看完这篇文章后也可以把流程自己讲出来,并且每个环节还可以讲出很多细节 他的消息机制离不开Looper.MessageQ ...
- Java方法:命令行传参,重载,可变参数,递归
Java方法:System.out.println()//系统类.out对象.输出方法Java方法是语句的集合,他们在一起执行一个功能方法是解决一类问题的步骤的有序组合方法包含于类或对象中方法在程序中 ...
- 鸿蒙第三方组件——SwipeCaptcha滑动拼图验证组件
目录:1.组件效果展示2.Sample解析3.<鸿蒙第三方组件>系列文章合集 前言 基于安卓平台的滑动拼图验证组件SwipeCaptcha( https://github.com/mcxt ...
- sitemesh简单介绍
SiteMesh 是一个网页布局和修饰的框架,利用它可以将网页的内容和页面结构分离,以达到页面结构共享的目的. Sitemesh是由一个基于Web页面布局.装饰以及与现存Web应用整合的框架. 它能帮 ...
- Hystrix 实战经验分享
一.背景 Hystrix是Netlifx开源的一款容错框架,防雪崩利器,具备服务降级,服务熔断,依赖隔离,监控(Hystrix Dashboard)等功能. 尽管说Hystrix官方已不再维护,且有A ...