P1559 运动员最佳匹配问题[最大费用最大流]
题目描述
羽毛球队有男女运动员各n人。给定2 个n×n矩阵P和Q。P[i][j]是男运动员i和女运动员j配对组成混合双打的男运动员竞赛优势;Q[i][j]是女运动员i和男运动员j配合的女运动员竞赛优势。由于技术配合和心理状态等各种因素影响,P[i][j]不一定等于Q[j][i]。男运动员i和女运动员j配对组成混合双打的男女双方竞赛优势为P[i][j]*Q[j][i]。设计一个算法,计算男女运动员最佳配对法,使各组男女双方竞赛优势的总和达到最大。
解析
这道题n很小,爆搜拿满分不成问题,但是本人不喜欢爆搜,还经常写错。
显然我们要求的是一个二分图最大带权匹配,最大费用最大流求之,spfa最长路+EK最大流。
当然,这道题明显是一个完备匹配,KM算法也可以,但是我没学。
注意细节,建费用边时,反向边设置边权为正向边的相反数。
感性理解一下,首先要明白增广路径算法中路径是可以反悔的,意思就是一条路搜到不合法的时候还可以沿着反向边回溯而答案不受影响,所以我们在反悔时就要用费用的相反数抵消掉影响。
参考代码
#include<cstdio>
#include<iostream>
#include<cmath>
#include<cstring>
#include<ctime>
#include<cstdlib>
#include<algorithm>
#include<queue>
#include<set>
#include<map>
#define N 51
#define M 10010
#define INF 0x3f3f3f3f
using namespace std;
inline int read()
{
int f=1,x=0;char c=getchar();
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
return x*f;
}
struct rec{
int next,ver,leng,edge;
}g[M<<1];
int head[M],tot=1,d[M],n,p[N][N],q[N][N],now[N],pre[N];
int ans;
bool v[M];
inline void add(int x,int y,int val,int e)
{
g[++tot].ver=y,g[tot].leng=val,g[tot].edge=e;
g[tot].next=head[x],head[x]=tot;
}
inline bool spfa()
{
fill(d,d+N,-INF);
memset(v,0,sizeof(v));
queue<int> q;
v[0]=1;d[0]=0;now[0]=INF;q.push(0);
while(q.size()){
int x=q.front();q.pop();
v[x]=0;
for(int i=head[x];i;i=g[i].next){
int y=g[i].ver,z=g[i].edge,leng=g[i].leng;
if(!leng||d[y]>=d[x]+z) continue;
d[y]=d[x]+z;
now[y]=min(now[x],leng);
pre[y]=i;
if(!v[y]) v[y]=1,q.push(y);
}
}
if(d[2*n+1]==-INF) return 0;
else return 1;
}
inline void change()
{
ans+=d[2*n+1]*now[2*n+1];
int tmp=2*n+1;
while(tmp!=0){
int i=pre[tmp];
g[i].leng-=now[2*n+1];
g[i^1].leng+=now[2*n+1];
tmp=g[i^1].ver;
}
}
int main()
{
n=read();
for(int i=1;i<=n;++i) add(0,i,1,0),add(i,0,0,0),add(i+n,n*2+1,1,0),add(n*2+1,i+n,0,0);
for(int i=1;i<=n;++i)
for(int j=1;j<=n;++j)
p[i][j]=read();
for(int i=1;i<=n;++i)
for(int j=1;j<=n;++j)
q[i][j]=read();
for(int i=1;i<=n;++i)
for(int j=1;j<=n;++j)
add(i,j+n,1,p[i][j]*q[j][i]),add(j+n,i,0,-p[i][j]*q[j][i]);
while(spfa()) change();
cout<<ans<<endl;
return 0;
}
P1559 运动员最佳匹配问题[最大费用最大流]的更多相关文章
- [洛谷 P1559] 运动员最佳匹配问题
题目描述 羽毛球队有男女运动员各n人.给定2 个n×n矩阵P和Q.P[i][j]是男运动员i和女运动员j配对组成混合双打的男运动员竞赛优势:Q[i][j]是女运动员i和男运动员j配合的女运动员竞赛优势 ...
- P1559 运动员最佳匹配问题
题目描述 羽毛球队有男女运动员各n人.给定2 个n×n矩阵P和Q.P[i][j]是男运动员i和女运动员j配对组成混合双打的男运动员竞赛优势:Q[i][j]是女运动员i和男运动员j配合的女运动员竞赛优势 ...
- 【题解】P1559 运动员最佳匹配问题
[题目](https://www.luogu.com.cn/problem/P1559) 题目描述 羽毛球队有男女运动员各n人.给定2 个n×n矩阵P和Q.P[i][j]是男运动员i和女运动员j配对组 ...
- 洛谷p1559运动员最佳匹配问题
题目 搜索 可行性剪枝 虽然这题目是我搜二分图的标签搜到的 但是n比较小 明显可以暴力 然而只有80分 再加上可行性剪纸就行啦 就是记所有运动员他所能匹配到的最大值. 在我们搜索到第i层的时候 如果他 ...
- KM模板 最大权匹配(广搜版) Luogu P1559 运动员最佳匹配问题
KM板题: #include <bits/stdc++.h> using namespace std; inline void read(int &num) { char ch; ...
- P1559 运动员最佳匹配问题 by hyl 天梦
#include<iostream> using namespace std; int n; int maxx[21][21]; int lie[21]; int aa[21]; int ...
- Luogu 1559 运动员最佳匹配问题(带权二分图最大匹配)
Luogu 1559 运动员最佳匹配问题(带权二分图最大匹配) Description 羽毛球队有男女运动员各n人.给定2 个n×n矩阵P和Q.P[i][j]是男运动员i和女运动员j配对组成混合双打的 ...
- 运动员最佳匹配问题 KM算法:带权二分图匹配
题面: 羽毛球队有男女运动员各n人.给定2 个n×n矩阵P和Q.P[i][j]是男运动员i和女运动员j配对组成混合双打的男运动员竞赛优势:Q[i][j]是女运动员i和男运动员j配合的女运动员竞赛优势. ...
- [Luogu 1559]运动员最佳匹配问题
Description 题库链接 求 \(2\times N\) 个点的带权二分图最佳匹配. \(1\leq N\leq 20\) Solution 我还是太菜了啊...到现在才学 \(KM\) . ...
随机推荐
- Spring boot后台搭建二为Shiro权限控制添加缓存
在添加权限控制后,添加方法 查看 当用户访问”获取用户信息”.”新增用户”和”删除用户”的时,后台输出打印如下信息 , Druid数据源SQL监控 为了避免频繁访问数据库获取权限信息,在Shiro中加 ...
- Linux服务器安装rocketMQ单机消息队列
首先下载rocketMQ 1.解压: > unzip rocketmq-all-4.3.0-source-release.zip > cd rocketmq-all-4.3.0/ > ...
- 基于 appium 的 UI 自动化测试
其中主要的目录和文件为: /MPTestCases ----------- 存放测试用例 /errorScreenShot ------------ 用例执行失败生成的错误截图 startTest.p ...
- django settings.py 配置文件
目录 settings.py 配置文件 settings.py 配置文件 import os BASE_DIR = os.path.dirname(os.path.dirname(os.path.ab ...
- 06-switch语句
switch语句 switch是一个条件语句,它是可以代替多个if else的常用方式 例子 package main import "fmt" func main() { a:= ...
- Go语言( 函数)
函数是组织好的.可重复使用的.用于执行指定任务的代码块.本文介绍了Go语言中函数的相关内容. 函数 Go语言中支持函数.匿名函数和闭包,并且函数在Go语言中属于“一等公民”. 函数定义 Go语言中定义 ...
- Linux进程的五个段
目录 数据段 代码段 BSS段 堆(heap) 栈 数据段 用来存放可执行文件中已初始化的全局变量,换句话说就是存放程序静态分配的变量和全局变量: 代码段 代码段是用来存放可执行文件的操作指令,也就是 ...
- tsql获取sqlserver某个库下所有表
) declare my_cursor cursor for (select [name] from SysObjects where XType='U') open my_cursor fetch ...
- 内网Https 自签Https证书 配合Tomcat 实现内网Https详细图文
内网项目启用Https配置手册 软件需求: OpenSSL https://www.openssl.org/ 已经安装了Java Jdk环境 制作前的需求: 已经配置了Jdk环境变量 安装好OpenS ...
- Unity的学习笔记(UGUI文本逐个字输出)
之前在网上找过各种的逐个输出字,我可能理解能力不好,照着代码复制没有能使用成功,于是自己研究了很多网上说的思路,各种改良出了一个能用的,写完自己测试,觉得还真好用,于是记录下来 用法:将用代码组件挂上 ...