HDU-2255 奔小康赚大钱
来源:互联网 发布:渣优化 编辑:程序博客网 时间:2024/05/06 21:25
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2255
解题思路:
二分图最优匹配的裸题,需要学习一下KM算法。。这道题也可以用网络流做,等学了网络流之后再写一下网络流的解题思路。
本题提供2个版本,一个是最朴素的KM算法,一个是优化后的KM算法(另外使用了输入外挂,成功刷入杭电前三)
代码如下:
#include<iostream>#include<cstdio>#include<cstring>#include<climits>#include<algorithm>using namespace std;#define N 310int map[N][N];bool visitx[N], visity[N];int lx[N], ly[N];int match[N];int n;bool Hungary(int u) //匈牙利算法{visitx[u] = true;for(int i = 0; i < n; ++i){if(!visity[i] && lx[u] + ly[i] == map[u][i]){visity[i] = true;if(match[i] == -1 || Hungary(match[i])){match[i] = u;return true;}}}return false;}void KM_perfect_match(){int temp;memset(lx, 0, sizeof(lx)); //初始化顶标memset(ly, 0, sizeof(ly)); //ly[i]为0for(int i = 0; i < n; ++i) //lx[i]为权值最大的边for(int j = 0; j < n; ++j)lx[i] = max(lx[i], map[i][j]);for(int i = 0; i < n; ++i) //对n个点匹配{while(1){memset(visitx, false, sizeof(visitx));memset(visity, false, sizeof(visity));if(Hungary(i)) //匹配成功break;else //匹配失败,找最小值{temp = INT_MAX;for(int j = 0; j < n; ++j) //x在交错树中if(visitx[j])for(int k = 0; k < n; ++k) //y在交错树外if(!visity[k] && temp > lx[j] + ly[k] - map[j][k])temp = lx[j] + ly[k] - map[j][k];for(int j = 0; j < n; ++j) //更新顶标{if(visitx[j])lx[j] -= temp;if(visity[j])ly[j] += temp;}}}}}int main(){int ans;while(scanf("%d", &n) != EOF){ans = 0;memset(match, -1, sizeof(match));for(int i = 0; i < n; ++i)for(int j = 0; j < n; ++j)scanf("%d", &map[i][j]);KM_perfect_match();for(int i = 0; i < n; ++i) //权值相加ans += map[match[i]][i];printf("%d\n", ans);}return 0;}
优化版本:
#include<iostream>#include<cstdio>#include<cstring>#include<climits>#include<algorithm>using namespace std;#define N 310int map[N][N];bool visitx[N], visity[N];int lx[N], ly[N];int slack[N];int match[N];int n;int Scan(){int res = 0 , ch ;while( !( ( ch = getchar() ) >= '0' && ch <= '9' ) ){if( ch == EOF ) return 1 << 30 ;}res = ch - '0' ;while( ( ch = getchar() ) >= '0' && ch <= '9' )res = res * 10 + ( ch - '0' ) ;return res ;}bool Hungary(int u) //匈牙利算法{visitx[u] = true;for(int i = 0; i < n; ++i){if(visity[i])continue;if(lx[u] + ly[i] == map[u][i]){visity[i] = true;if(match[i] == -1 || Hungary(match[i])){match[i] = u;return true;}}else //不在相等子图slack[i] = min(slack[i], lx[u] + ly[i] - map[u][i]);}return false;}void KM_perfect_match(){int temp;memset(lx, 0, sizeof(lx)); //初始化顶标memset(ly, 0, sizeof(ly)); //ly[i]为0for(int i = 0; i < n; ++i) //lx[i]为权值最大的边for(int j = 0; j < n; ++j)lx[i] = max(lx[i], map[i][j]);for(int i = 0; i < n; ++i) //对n个点匹配{for(int j = 0; j < n; ++j)slack[j] = INT_MAX;while(1){memset(visitx, false, sizeof(visitx));memset(visity, false, sizeof(visity));if(Hungary(i)) //匹配成功break;else //匹配失败,找最小值{temp = INT_MAX;for(int j = 0; j < n; ++j)if(!visity[j])if(temp > slack[j])temp = slack[j];for(int j = 0; j < n; ++j) //更新顶标{if(visitx[j])lx[j] -= temp;if(visity[j])ly[j] += temp;elseslack[j] -= temp;}}}}}int main(){int ans;while(scanf("%d", &n) != EOF){ans = 0;memset(match, -1, sizeof(match));for(int i = 0; i < n; ++i)for(int j = 0; j < n; ++j)map[i][j] = Scan();KM_perfect_match();for(int i = 0; i < n; ++i) //权值相加ans += map[match[i]][i];printf("%d\n", ans);}return 0;}
- HDU-2255 奔小康赚大钱
- HDU 2255 奔小康赚大钱
- hdu(2255)奔小康赚大钱
- hdu 2255 奔小康赚大钱
- HDU 2255 奔小康赚大钱
- 奔小康赚大钱 hdu 2255
- HDU 2255奔小康赚大钱
- HDU-2255 奔小康赚大钱
- [HDU]2255-奔小康赚大钱
- HDU 2255 奔小康赚大钱
- HDU 2255 奔小康赚大钱
- 奔小康赚大钱 HDU
- 奔小康赚大钱 HDU
- HDU 2255 奔小康赚大钱 KM算法
- hdu 2255 KM 奔小康赚大钱
- hdu 2255 奔小康赚大钱 (KM裸题)
- HDU 2255 奔小康赚大钱(KM模板)
- HDU-2255 奔小康赚大钱 裸KM
- QT VS2008开发环境
- 2012年的几个任务
- 网络的经济基础篇-之三-AH、ESP
- Linux下修改SHELL字体背景颜色笔记
- FCFS,高优先权算法进程调度程序模拟
- HDU-2255 奔小康赚大钱
- 我的2011
- LTE开机到数据传输的过程学习
- PerfHud安装、初次使用
- struts2多配置文件使用通配符加载
- Myeclipse 的汉化
- CentOS 5.5升级内核到2.6.28并支持XFS/L7
- ARM linux的启动部分源代码简略分析
- 英雄