题意:

      给你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 最大匹配可能性的更多相关文章

  1. POJ1904 强联通(最大匹配可能性)

    题意:       有n个王子,n个公主,然后给你每个王子喜欢的公主,最后问你在不影响最大匹配的前提下,每个王子可以匹配那些公主. 思路:       是hdu4685的减弱版,之前研究过hdu468 ...

  2. UVALive 5033 I'm Telling the Truth 二分图最大匹配(略有修改)

    I - I'm Telling the Truth Time Limit:3000MS     Memory Limit:0KB     64bit IO Format:%lld & %llu ...

  3. Python学习实践------正向最大匹配中文分词

    正向最大匹配分词: 1.加载词典文件到集合中,取词典文件中最大长度词的length 2.每次先在句子中按最大长度分割,然后判断分割的词是否存在字典中,存在则记录此词,调整起始点. 3.不存在则按最大长 ...

  4. UOJ79 一般图最大匹配

    题目描述 从前一个和谐的班级,所有人都是搞OI的.有 nn 个是男生,有 00 个是女生.男生编号分别为 1,-,n1,-,n. 现在老师想把他们分成若干个两人小组写动态仙人掌,一个人负责搬砖另一个人 ...

  5. hdu 1281 二分图最大匹配

    对N个可以放棋子的点(X1,Y1),(x2,Y2)......(Xn,Yn);我们把它竖着排看看~(当然X1可以对多个点~) X1   Y1 X2   Y2 X3   Y3 ..... Xn   Yn ...

  6. POJ 2226二分图最大匹配

    匈牙利算法是由匈牙利数学家Edmonds于1965年提出,因而得名.匈牙利算法是基于Hall定理中充分性证明的思想,它是二部图匹配最常见的算法,该算法的核心就是寻找增广路径,它是一种用增广路径求二分图 ...

  7. hdu-1179-二分图最大匹配

    Ollivanders: Makers of Fine Wands since 382 BC. Time Limit: 2000/1000 MS (Java/Others)    Memory Lim ...

  8. codevs1022 覆盖[Hungary 二分图最大匹配]

    codevs1022 覆盖 有一个N×M的单位方格中,其中有些方格是水塘,其他方格是陆地.如果要用1×2的矩阵区覆盖(覆盖过程不容许有任何部分重叠)这个陆地,那么最多可以覆盖多少陆地面积. 输入描述  ...

  9. POJ1274 The Perfect Stall[二分图最大匹配]

    The Perfect Stall Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 23911   Accepted: 106 ...

随机推荐

  1. Java语言的魅力

    Java语言的简介 Java是一门面向对象编程语言,不仅吸收了C++语言的各种优点,还摒弃了C++里难以理解的多继承.指针等概念,因此Java语言具有功能强大和简单易用两个特征.Java语言作为静态面 ...

  2. 让人头疼的AI bug (随想)

    虽然概念上,人工智能和机器学习不等同.但是本文提及的AI,指的是基于机器学习的AI.   一个软件产品,出了错误叫bug,bug需要修.那一个机器学习的模型,准确率在那摆着呢,大伙心知肚明是有一定的犯 ...

  3. java 给时间增加时间得到一个新的时间(日期)

    SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd") LocalDate expirationDate String exp ...

  4. Linux系统用户与用户组管理

    一.用户和用户组的管理 1.新增组 groupadd 命令 格式:groupadd 组名 2.删除组 groupdel 格式:groupdel 组名 3.增加用用户命令 useradd   格式:us ...

  5. P3388 【模板】割点(割顶) 题解 (Tarjan)

    题目链接 P3388 [模板]割点(割顶) 解题思路 最近学的东西太杂了,多写点博客免得自己糊里糊涂的过去了. 这个题求割点,感觉这篇文章写得挺好. 割点是啥?如果去掉这个点之后连通图变成多个不连通图 ...

  6. 题解 洛谷P1990 覆盖墙壁

    DP康复训练题 原题:洛谷P1990 核心:递推/DP 题源应该是铺地砖,所以采用一摸一样的思路,只是有两种不同的方块 我们先用最最简单的方式尝试一下枚举当最后一行被填满的情况: 1.如果我们只用第一 ...

  7. Spring 声明式事务与编程式事务详解

    本文转载自IBM开发者论坛:https://developer.ibm.com/zh/articles/os-cn-spring-trans 根据自己的学习理解有所调整,用于学习备查. 事务管理对于企 ...

  8. Flutter原理简介

    Flutter 是怎么运转的? 与用于构建移动应用程序的其他大多数框架不同,Flutter 是重写了一整套包括底层渲染逻辑和上层开发语言的完整解决方案.这样不仅可以保证视图渲染在 Android 和 ...

  9. abp加DDD开发:低耦合、可复用、可扩展的【工单】业务模块-简介和集成

    前言 很多场景[单体+模块化]比微服务更合适,开发难度低.代码可复用性强.可扩展性强.模块化开发有些难点,模块启动与卸载.模块之间的依赖和通讯.asp.net core abp为我们提供了模块化开发能 ...

  10. Myabtis-Plus之QueryWrapper常用方法

    AbstractWrapper 下的方法及使用 方法名 说明 使用 allEq(Map<R, V> params) 全部 =(或个别 isNull) allEq(params,true) ...