UVa 1045 - The Great Wall Game 最小权匹配


题意:给你一个n*n的棋盘,上面有n个棋子,要求通过移动各个棋子使得棋子在同一行或者同一列或者对角线上,求最小移动次数。

思路:直接对于所有可能情况构造二分图,X集合为最初棋子,Y集合为移动后的棋子方位,边权为移动的次数。然后KM算法求最小权匹配。

/* **********************************************
Author : JayYe
Created Time: 2013-8-18 15:55:41
File Name : zzz.cpp
*********************************************** */ #include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std; int max(int a, int b) { return a>b?a:b; }
int min(int a, int b) { return a>b?b:a; } const int maxn = 22; struct PP {
int x, y;
}a[maxn]; int n, slack[maxn], lx[maxn], ly[maxn], match[maxn], w[maxn][maxn];
bool S[maxn], T[maxn]; bool dfs(int i) {
S[i] = 1;
for(int j = 1;j <= n; j++) if(!T[j])
slack[j] = min(slack[j], w[i][j] - lx[i] - ly[j]);
for(int j = 1;j <= n; j++) if(w[i][j] == lx[i] + ly[j] && !T[j]) {
T[j] = 1;
if(!match[j] || dfs(match[j])) {
match[j] = i;
return true;
}
}
return false;
} void update() {
int delta = 1<<22;
for(int i = 1;i <= n; i++) if(!T[i])
delta = min(delta, slack[i]);
for(int i = 1;i <= n; i++) {
if(S[i]) lx[i] += delta;
if(T[i]) ly[i] -= delta;
}
} void KM() {
int i, j;
for(i = 1;i <= n; i++) {
ly[i] = match[i] = 0;
lx[i] = 1<<22;
for(j = 1;j <= n; j++)
lx[i] = min(lx[i], w[i][j]);
}
for(i = 1;i <= n; i++) {
while(true) {
for(j = 1;j <= n; j++) S[j] = T[j] = 0, slack[j] = 1<<22;
if(dfs(i)) break;
else update();
}
}
} int solve() {
int i, j, k; for(i = 1;i <= n; i++)
scanf("%d%d" ,&a[i].x, &a[i].y);
int ans = 1<<22;
// 棋子在同一行的情况
for(i = 1;i <= n; i++) {
for(j = 1;j <= n; j++)
for(k = 1;k <= n; k++)
w[j][k] = abs(i - a[j].x) + abs(k - a[j].y);
KM();
int cur = 0;
for(j = 1;j <= n; j++) cur += lx[j] + ly[j];
ans = min(ans, cur);
}
// 棋子在同一列的情况
for(i = 1;i <= n; i++) {
for(j = 1;j <= n; j++)
for(k = 1;k <= n; k++)
w[j][k] = abs(k - a[j].x) + abs(i - a[j].y);
KM();
int cur = 0;
for(j = 1;j <= n; j++) cur += lx[j] + ly[j];
ans = min(ans, cur);
}
// 棋子在对角线的两种情况
for(i = 1;i <= n; i++)
for(j = 1;j <= n; j++)
w[i][j] = abs(j - a[i].x) + abs(j - a[i].y);
KM();
int cur = 0;
for(i = 1;i <= n; i++) cur += lx[i] + ly[i];
ans = min(ans, cur); for(i = 1;i <= n; i++)
for(j =1;j <= n; j++)
w[i][j] = abs(j - a[i].x) + abs(n-j+1 - a[i].y);
KM();
cur = 0;
for(i = 1;i <= n; i++) cur += lx[i] + ly[i];
ans = min(ans, cur);
return ans;
} int main() {
int cas = 1;
while(scanf("%d", &n) != -1 && n) {
printf("Board %d: %d moves required.\n\n", cas++, solve());
}
return 0;
}

UVa 12168 - Cat vs. Dog 最大独立集

根据题意直接构造二分图,X集合表示喜欢狗的人,Y集合表示喜欢猫的人,如果X集合里的人不喜欢某只猫,就与Y集合里喜欢该猫的人连边,反之也一样。那么要使得尽可能多的人满足愿望,也就是求最大独立集。

最大独立集 = N - 最大匹配

/* **********************************************
Author : JayYe
Created Time: 2013-8-18 16:53:58
File Name : zzz.cpp
*********************************************** */ #include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std; const int maxn = 500+10;
struct PP {
int x, y;
PP() {}
PP(int x, int y) : x(x), y(y) {}
}cat[maxn], dog[maxn]; int n, m, match[maxn];
bool vis[maxn], mp[maxn][maxn]; bool dfs(int i) {
for(int j = 1;j <= m; j++) if(mp[i][j] && !vis[j]) {
vis[j] = 1;
if(!match[j] || dfs(match[j])) {
match[j] = i;
return true;
}
}
return false;
} int main() {
int i, j, t, c, d;
scanf("%d", &t);
while(t--) {
scanf("%d%d%d", &c, &d, &n);
int n1 = 0, n2 = 0, x, y;
for(i = 1;i <= n; i++) {
char ch1[11], ch2[11];
scanf("%s%s", ch1, ch2);
if(ch1[0] == 'C') {
sscanf(ch1+1, "%d", &x);
sscanf(ch2+1, "%d", &y);
cat[++n1] = PP(x, y);
}
else {
sscanf(ch1+1, "%d", &x);
sscanf(ch2+1, "%d", &y);
dog[++n2] = PP(x, y);
}
}
n = n1, m = n2;
if(n == 0 || m == 0) {
printf("%d\n", n+m); continue;
}
for(i = 1;i <= n; i++) {
for(j = 1;j <= m; j++) {
if(cat[i].y == dog[j].x || cat[i].x == dog[j].y)
mp[i][j] = 1;
else
mp[i][j] = 0;
}
}
for(i = 1;i <= m; i++) match[i] = 0;
int ans = 0;
for(i = 1;i <= n; i++) {
for(j = 1;j <= m; j++) vis[j] = 0;
if(dfs(i)) ans++;
}
printf("%d\n", n+m - ans);
}
return 0;
}

Uva 1349 - Optimal Bus Route Design

题意:给你一个n个点的有向带环图,要你找出几个圈使得每个结点只属于一个圈,要求输出最小的总的长度,如果没有这样的方案,输出N。

构造二分图,把所有的结点拆成两个,放在X集合的为i, 放在Y集合的为 i ',如果有边i -> j,则在图中引入边i -> j',这样子构造好后实际上就是求最小权完美匹配,如果没有完美匹配则无解。

这里需要注意的是输入的边可能有好多条是重复的但是权值不同,需要取最小权,这个wa了我好几发。。。

/* **********************************************
Author : JayYe
Created Time: 2013-8-18 17:31:11
File Name : zzz.cpp
*********************************************** */ #include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std; int max(int a, int b) { return a>b?a:b; }
int min(int a, int b) { return a>b?b:a; } const int INF = 1<<22;
const int maxn = 100+10;
int n, slack[maxn], match[maxn], lx[maxn], ly[maxn], w[maxn][maxn];
bool S[maxn], T[maxn]; bool dfs(int i) {
S[i] = 1;
for(int j = 1;j <= n; j++) if(!T[j])
slack[j] = min(slack[j], w[i][j] - lx[i] - ly[j]);
for(int j = 1;j <= n; j++) if(w[i][j] == lx[i] + ly[j] && !T[j]) {
T[j] = 1;
if(!match[j] || dfs(match[j])) {
match[j] = i;
return true;
}
}
return false;
} void update() {
int delta = INF;
for(int i = 1;i <= n; i++) if(!T[i])
delta = min(delta, slack[i]);
for(int i = 1;i <= n; i++) {
if(S[i]) lx[i] += delta;
if(T[i]) ly[i] -= delta;
}
} void KM() {
int i, j;
for(i = 1;i <= n; i++) {
ly[i] = match[i] = 0;
lx[i] = INF;
for(j = 1;j <= n; j++)
lx[i] = min(lx[i], w[i][j]);
} for(i = 1;i <= n; i++) {
while(true) {
for(j = 1;j <= n; j++) S[j] = T[j] = 0, slack[j] = INF;
if(dfs(i)) break;
else update();
}
}
} void solve() {
int i, j;
for(i = 1;i <= n; i++)
for(j = 1;j <= n; j++)
w[i][j] = INF;
for(i = 1;i <= n; i++) {
while(scanf("%d", &j) && j) {
int dis;
scanf("%d", &dis);
// 同一条路要取最小值
w[i][j] = min(w[i][j], dis);
}
}
KM();
int ans = 0;
for(i = 1;i <= n; i++) ans += lx[i] + ly[i];
if(ans > INF-10) puts("N");
else printf("%d\n", ans);
} int main() {
while(scanf("%d", &n) != -1 && n) {
solve();
}
return 0;
}

UVa 二分图匹配 Biginners的更多相关文章

  1. UVa 二分图匹配 Examples

    这些都是刘汝佳的算法训练指南上的例题,基本包括了常见的几种二分图匹配的算法. 二分图是这样一个图,顶点分成两个不相交的集合X , Y中,其中同一个集合中没有边,所有的边关联在两个集合中. 给定一个二分 ...

  2. POJ 2289 Jamie's Contact Groups / UVA 1345 Jamie's Contact Groups / ZOJ 2399 Jamie's Contact Groups / HDU 1699 Jamie's Contact Groups / SCU 1996 Jamie's Contact Groups (二分,二分图匹配)

    POJ 2289 Jamie's Contact Groups / UVA 1345 Jamie's Contact Groups / ZOJ 2399 Jamie's Contact Groups ...

  3. uva 12083 Guardian of Decency (二分图匹配)

    uva 12083 Guardian of Decency Description Frank N. Stein is a very conservative high-school teacher. ...

  4. UVA 12549 - 二分图匹配

    题意:给定一个Y行X列的网格,网格种有重要位置和障碍物.要求用最少的机器人看守所有重要的位置,每个机器人放在一个格子里,面朝上下左右四个方向之一发出激光直到射到障碍物为止,沿途都是看守范围.机器人不会 ...

  5. UVA 1663 Purifying Machine (二分图匹配,最大流)

    题意: 给m个长度为n的模板串,模板串由0和1和*三种组成,且每串至多1个*,代表可0可1.模板串至多匹配2个串,即*号改成0和1,如果没有*号则只能匹配自己.问:模板串可以缩减为几个,同样可以匹配原 ...

  6. UVA 11045-My T-shirt suits me(二分图匹配)

    题意:有N件T恤,N是6的倍数,因为有6种型号,每种件数相同,有M个人,每个人有两种型号的T恤适合他,每个人可以挑其中的一种,问能否所有的人都能分配到T恤. 解析:典型的二分图匹配,每N/6为同种T恤 ...

  7. POJ 1274 裸二分图匹配

    题意:每头奶牛都只愿意在她们喜欢的那些牛栏中产奶,告诉每头奶牛愿意产奶的牛棚编号,求出最多能分配到的牛栏的数量. 分析:直接二分图匹配: #include<stdio.h> #includ ...

  8. BZOJ1433 ZJOI2009 假期的宿舍 二分图匹配

    1433: [ZJOI2009]假期的宿舍 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2375  Solved: 1005[Submit][Sta ...

  9. HDU1281-棋盘游戏-二分图匹配

    先跑一个二分图匹配,然后一一删去匹配上的边,看能不能达到最大匹配数,不能这条边就是重要边 /*----------------------------------------------------- ...

随机推荐

  1. Android之判断设备网络连接状态,并判断连接方式

    在Android开发过程中,对于一个需要连接网络的Android设备,对设备的网络状态检测是很有必要的!有很多的App都需要连接网络.判断设备是否已经连接网络,并且在连接网络的状态下判断是wifi无线 ...

  2. ThinkPHP第六天(正则表达式)

    1.正则表达式:原子 ①a-z,A-Z,0-9,_ ②用(abc)圆括号括起来的单元符号,表示括号里面的作为一个完整的组合,必须完成匹配,不被拆分来 ③用方括号括起来[abc][^abc]称之为原子表 ...

  3. float与position

    使用float会使块级元素的宽高表现为包裹内容(在不设定宽高的情况下)  这是当然的  我们使用float就是使几个div排在一行 当然不可能在宽度上撑满父元素啦  至于高度 不论有没有float 高 ...

  4. 深入探究VC —— 编译器cl.exe(1)

    cl.exe的功能是将源代码文件编译为可提供链接器使用的obj对象文件.cl.exe命令行参数形式如下: CL (option...) file... [option | file]... [lib. ...

  5. HDU p1294 Rooted Trees Problem 解题报告

    http://www.cnblogs.com/keam37/p/3639294.html keam所有 转载请注明出处 Problem Description Give you two definit ...

  6. Python 連接 MySQL

    Python 連接 MySQL MySQL 是十分流行的開源資料庫系統,很多網站也是使用 MySQL 作為後台資料儲存,而 Python 要連接 MySQL 可以使用 MySQL 模組.MySQLdb ...

  7. PHP 自学之路-----XML编程(Xpath技术,simpleXml技术)基础入门

    XPAth技术 XPath的设计的核心思想,可以通过xpath迅速简介的定位到你希望查找的节点.主要目的是描述节点相对其他节点的位置,可以取得所有符合条件的节点,成为[位置路径]. Xapth主要用来 ...

  8. POJ 2455 Secret Milking Machine(搜索-二分,网络流-最大流)

    Secret Milking Machine Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 9658   Accepted: ...

  9. eclipse+python+Anaconda ML环境搭建

    整个流程按照这个来: http://blog.csdn.net/olaking/article/details/42435451 你会发现pydev安装不了,听说服务器被和谐了,下面这篇文章告诉你如何 ...

  10. CodeForces 486C Palindrome Transformation 贪心+抽象问题本质

    题目:戳我 题意:给定长度为n的字符串,给定初始光标位置p,支持4种操作,left,right移动光标指向,up,down,改变当前光标指向的字符,输出最少的操作使得字符串为回文. 分析:只关注字符串 ...