意甲冠军:今天,有n王子,m公主。现在给他们配对,与王子会嫁给一个男人,他喜欢。公主无法做出选择。

这标题去咬硬,还有一类似的题目poj1904。那个题目也是给王子与公主配对,但那个是王子公主各n个,且给定了一个完美匹配,然后求每一个王子能够做出的选择且不影响最大匹配数目。那题是先建各条喜欢关系的边。然后在由被选择的公主连一条边到与之配对的王子。强连通之后假设一个王子和一个公主在一个强连通分量中,那么他们结合的话,他们的还有一半也各自能在强连通中找到另外的匹配,就是符合题意的结果了。

这个题目算是升级版把。我们须要做的先是用匈牙利算法求出最大匹配res,然后建立m-res个虚拟王子与m-res单身公主准备匹配,建立n-res个虚拟公主与n-res个单身王子准备匹配。过程就是虚拟王子要喜欢每个公主,相同虚拟公主也要被每个王子喜欢,这样最大匹配一定是n+m-res.求出一个这种完美匹配,然后再套用poj1904的思路用强连通做。建议先做下1904额。

代码:

#include<iostream>
#include<cstdio>
#include<string>
#include<cmath>
#include<queue>
#include<stack>
#include<map>
#include<cstring>
#include<algorithm>
#define rep(i,a,b) for(int i=(a);i<(b);i++)
#define rev(i,a,b) for(int i=(a);i>=(b);i--)
#define clr(a,x) memset(a,x,sizeof a)
#define inf 0x3f3f3f3f
typedef long long LL;
using namespace std; const int eps=0.00000001;
const int maxn=2005;
const int maxm=maxn*maxn/2; int first[maxn],link[maxn];
int nex[maxm],w[maxm],v[maxm],u[maxm];
bool done[maxn],g[maxn][maxn];
int n,m,ecnt; void add_(int a,int b,int c=0)
{
u[ecnt]=a;
v[ecnt]=b;
w[ecnt]=c;
nex[ecnt]=first[a];
first[a]=ecnt++;
}
bool dfs(int s)
{
for(int e=first[s];~e;e=nex[e])
if(!done[v[e]])
{
done[v[e]]=true;
if(link[v[e]]==-1||dfs(link[v[e]]))
{
link[v[e]]=s;
return true;
}
}
return 0;
} int hungary(int n)
{
int ans=0;
clr(link,-1);
for(int i=1;i<=n;i++)
{
clr(done,false);
if(dfs(i))ans++;
}
return ans;
}
int low[maxn],dfn[maxn],stck[maxn],belong[maxn];
int index,top,scc;
bool ins[maxn];
int num[maxn];
int in[maxn],out[maxn]; void tarjan(int u)
{
low[u]=dfn[u]=++index;
stck[top++]=u;
ins[u]=1;
for(int e=first[u];~e;e=nex[e])
{
if(!dfn[v[e]])
{
tarjan(v[e]);
low[u]=min(low[u],low[v[e]]);
}
else if(ins[v[e]])low[u]=min(low[u],dfn[v[e]]);
}
if(low[u]==dfn[u])
{
int v;
scc++;
do
{
v=stck[--top];
ins[v]=false;
belong[v]=scc;
num[scc]++;
}while(v!=u);
}
}
void solve(int n)
{
clr(dfn,0);
clr(ins,0);
clr(num,0);
index=scc=top=0;
for(int i=1;i<=n;i++)
if(!dfn[i])tarjan(i);
} int main()
{
int t,a,b,c,k,cas=1,key=1000;
scanf("%d",&t);
while(t--)
{
clr(first,-1);ecnt=0;
clr(g,false);
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
{
scanf("%d",&k);
while(k--)
{
scanf("%d",&a);
if(!g[i][a])
{
g[i][a]=1;
add_(i,a+key);
}
}
}
int res=hungary(n);
int nn=n+m-res;
for(int i=n+1;i<=nn;i++)
for(int j=1;j<=nn;j++)
add_(i,j+key),g[i][j]=1;
for(int i=1;i<=n;i++)
for(int j=m+1;j<=nn;j++)
add_(i,j+key),g[i][j]=1;
hungary(nn); ecnt=0;clr(first,-1);
for(int i=1;i<=nn;i++)
if(link[i+key]!=-1)add_(i+nn,link[i+key]); for(int i=1;i<=nn;i++)
for(int j=1;j<=nn;j++)
if(g[i][j])add_(i,j+nn);
solve(2*nn);
printf("Case #%d:\n",cas++);
int ans[1000];
for(int i=1;i<=n;i++)
{
int en=0;
for(int j=1;j<=m;j++)
if(g[i][j]&&belong[j+nn]==belong[i])ans[en++]=j;
printf("%d",en);
for(int i=0;i<en;i++)
printf(" %d",ans[i]);
puts("");
}
}
return 0;
}

版权声明:本文博客原创文章。博客,未经同意,不得转载。

HDU4685 Prince and Princess 完美搭配+良好的沟通的更多相关文章

  1. 强连通+二分匹配(hdu4685 Prince and Princess)

    Prince and Princess Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Othe ...

  2. HDU4685 Prince and Princess【强连通】

    题意: 有n个王子和m个公主,每个王子都会喜欢若干个公主,也就是王子只跟自己喜欢的公主结婚,公主就比较悲惨, 跟谁结婚都行.然后输出王子可能的结婚对象,必须保证王子与任意这些对象中的一个结婚,都不会影 ...

  3. HDU4685:Prince and Princess(二分图匹配+tarjan)

    Prince and Princess Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Othe ...

  4. LESS-Middleware:Node.js 和 LESS 的完美搭配

    LESS 是一个编写 CSS 的很好的方式 ,让你可以使用变量,嵌套规则,混入以及其它许多有用的功能,它可以帮助您更好地组织你的 CSS 代码. 最近我一直在研究 Node.js ,并想用 less- ...

  5. 10635 - Prince and Princess

    Problem D Prince and Princess Input: Standard Input Output: Standard Output Time Limit: 3 Seconds In ...

  6. UVa10653.Prince and Princess

    题目连接:http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...

  7. uva 10635 - Prince and Princess(LCS)

    题目连接:10635 - Prince and Princess 题目大意:给出n, m, k,求两个长度分别为m + 1 和 k + 1且由1~n * n组成的序列的最长公共子序列长的. 解题思路: ...

  8. Prince and Princess HDU - 4685(匹配 + 强连通)

    Prince and Princess Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Othe ...

  9. UVA - 10635 Prince and Princess LCS转LIS

    题目链接: http://bak.vjudge.net/problem/UVA-10635 Prince and Princess Time Limit: 3000MS 题意 给你两个数组,求他们的最 ...

随机推荐

  1. Swift - 闭包的介绍及用法(以数组排序为例)

    闭包(即一些小的匿名代码块),可以像函数一样使用.可以很方便的将闭包传给其他函数,告诉它们应当如何执行某一个任务. 1,使用sort方法和闭包进行数组排序 sort方法返回一个数组的有序版本.(sor ...

  2. python模块介绍- binascii 二进制和ASCII转换

    python模块介绍-binascii二进制和ASCII转换 目录 项目简介 简介: Uu编码 Binhex编码 Base64编码 QP码 CRC校验和 二进制转换 其他实例 项目简介 Python中 ...

  3. 重操JS旧业第九弹:函数表达式

    函数表达式,什么概念,表达式中的函数表达式. 1 函数申明 function 函数名([函数参数]){ //函数体 } js中无论像这样的显示函数什么放在调用之前还是调用之后,都不影响使用,因为js解 ...

  4. struts2 <s:iterator/>怎样取得循环的索引

    <s:iterator value="list" id="user" status="L"> <s:property va ...

  5. 性能测试之LoardRunner 手动关联二

    概述: 1.如果寻找左右边界值 2.关联函数详解 以下是详细介绍 1.如果寻找左右边界值 <以login 为例> Step1.录制两份相同的业务流程的的脚本,输入的数据要相同 Step2. ...

  6. ZYNQ-7000 Unable to connect to ps7_cortexa9 解决方案

    图1 开发工具:Xilinx SDk 14.4(基于Eclipse,ISE suite 14.4组件之一) 开发板:Xilinx ZYNQ-7000 zc702 rev 1.0(注意:这个板子的版本说 ...

  7. struts2 一个简洁的struts.xml

    struts.xml <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE struts PUB ...

  8. OSGi:生命周期层

    前言 生命周期层在OSGi框架中属于模块层上面的一层,它的运作是建立在模块层的功能之上的.生命周期层一个主要的功能就是让你能够从外部管理应用或者建立能够自我管理的应用(或者两者的结合),并且给了应用本 ...

  9. 源代码编译lamp环境

    没有办法用 rpm查询一个源代码包是否安装 因为 并不是用rpm安装的 可以先吧 selinux 给禁用掉  iptables -F 把防火墙规则全部删除 首先确保 gcc  gcc-c++   ma ...

  10. regsvr32 命令小集注册OCX控件,注册控件(包括十几个举例)

    Regsvr32 进程文件: regsvr32 or regsvr32.exe  进程名称: Microsoft DLL Registration Service  英文描述: regsvr32.ex ...