http://poj.org/problem?id=1325
题目描述:
众所周知,机器调度是计算机科学中非常典型的一个问题,已经被研究很长时间了。各种机器调度问题在以下方面差别很大:必须满足的约束条件,以及期望得到的调度时间表。现在考虑一个针对2 台机器的机器调度问题。
假设有2台机器,A和B。机器A有n种工作模式,分别称为mode_0, mode_1, …, mode_n-1。同样机器B 有m 种工作模式,分别为mode_0,mode_1, … , mode_m-1。刚开始时,A 和B 都工作在模式mode_0。
给定k 个作业,每个作业可以工作在任何一个机器的特定模式下。例如,作业0 可以工作在机器A 的模式mode_3 或者机器B 的mode_4 模式,作业1 可以工作在机器A 的模式mode_2或者工作在机器B 的模式mode_4,等等。因此,对作业j,调度中的约束条件可以表述成一个三元组(i, x, y),意思是作业i 可以工作在机器A 的mode_x 模式或者机器B 的mode_y 模式。
很显然的是,为了完成所有的作业,必须时不时切换机器的工作模式,但不幸的是,机器工作模式的切换只能通过手动重启机器完成。编写程序实现:改变作业的顺序,给每个作业分配合适的机器,使得重启机器的次数最少。
分析:
首先构造二部图:把A 的n 个mode 和B 的m 个mode 看作图的顶点,如果某个任务可以在A 的mode_i 或B 的mode_j 上完成,则从Ai 到Bj 连接一条边,这样构造了一个二部图。
例如对题目样例输入中的测试数据,构造的二部图如图7.28 所示。
原先机器A 和机器B 都是工作在模式_0,切换到机器A 的模式_1,可以完成作业(0, 1, 1)、(1, 1, 2)、(2, 1, 3)、(3, 1, 4),再切换到机器A 的模式_2,可以完成作业(4, 2, 1)、(5, 2, 2)、(6, 2,3)、(7, 2, 4),最后切换到机器B 的模式_3,可以完成作业(8, 3, 3)、(9, 4, 3)。所以需要手动重启
机器3 次。
本题要求二部图的最小点覆盖集问题,即求最小的顶点集合,“覆盖”住所有的边。转换成求二部图的最大匹配问题,因为:二部图的点覆盖数α0 = 匹配数β1。另外,机器A 和机器B 最初工作在模式_0,所以对于那些可以工作在机器A 的模式_0 或者机器B 的模式_0 的作业,在完成这些作业时是不需要重启机器的。
692K 0MS G++
#include<iostream>
#include<cstring>
#include<cstdio>
#include<vector>
const int MAXN=110;
using namespace std;
int n,m,k;
vector<int> g[MAXN];
int from[MAXN],tot;
bool use[MAXN];
bool match(int x){
for(int i=0;i<g[x].size();++i){
if(!use[g[x][i]]){
use[g[x][i]]=true;
if(from[g[x][i]]==-1||match(from[g[x][i]])){
from[g[x][i]]=x;
return true;
}
}
}
return false;
}
int hungary(){
tot=0;
memset(from,0xff,sizeof(from));
for(int i=1;i<=n;++i){
memset(use,0,sizeof(use));
if(match(i))
tot++;
}
return tot;
}
int main()
{
int num,u,v;
while(scanf("%d",&n),n){
scanf("%d%d",&m,&k);
for(int i=0;i<=n;++i) g[i].clear();
for(int i=0;i<k;++i){
scanf("%d%d%d",&num,&u,&v);
if(u&&v) g[u].push_back(v);
}
printf("%d\n",hungary());
}
return 0;
}