POJ1325.Machine Schedule机器调度——最小顶点覆盖

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;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用EasyExcel导入Excel文件非常简单,只需要遵循以下步骤: 1. 在pom.xml文件中添加easyexcel依赖: ```xml <dependency> <groupId>com.alibaba</groupId> <artifactId>easyexcel</artifactId> <version>2.2.2</version> </dependency> ``` 2. 创建一个实体类来存储Excel表格中的数据。 ```java @Data public class ExcelData { private String name; private Integer age; private String email; } ``` 3. 创建一个Excel读取器类。 ```java public class ExcelReader { public static List<ExcelData> readExcel(String fileName) throws Exception { InputStream inputStream = new FileInputStream(fileName); ExcelReader excelReader = new ExcelReader(inputStream, null, new ExcelListener()); excelReader.read(); List<ExcelData> dataList = ((ExcelListener)excelReader.getEventListener()).getDataList(); inputStream.close(); return dataList; } } ``` 4. 创建一个Excel监听器类,用于处理Excel表格中的数据。 ```java public class ExcelListener extends AnalysisEventListener<ExcelData> { private List<ExcelData> dataList = new ArrayList<>(); @Override public void invoke(ExcelData excelData, AnalysisContext analysisContext) { dataList.add(excelData); } @Override public void doAfterAllAnalysed(AnalysisContext analysisContext) { } public List<ExcelData> getDataList() { return dataList; } public void setDataList(List<ExcelData> dataList) { this.dataList = dataList; } } ``` 5. 在Controller中调用ExcelReader类的readExcel方法来读取Excel文件中的数据。 ```java @RestController public class ExcelController { @RequestMapping("/importExcel") public String importExcel(@RequestParam("file") MultipartFile file) { try { String fileName = file.getOriginalFilename(); List<ExcelData> dataList = ExcelReader.readExcel(fileName); // 处理数据 return "导入成功"; } catch (Exception e) { e.printStackTrace(); return "导入失败"; } } } ``` 这样就可以使用EasyExcel导入Excel文件了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值