USACO Training 4.2.3 Job Processing 工序安排 题解与分析

2024-06-07 07:18

本文主要是介绍USACO Training 4.2.3 Job Processing 工序安排 题解与分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Job Processing 工序安排

IOI'96


描述

一家工厂的流水线正在生产一种产品,这需要两种操作:操作A和操作B。每个操作只有一些机器能够完成。

Ioi96d1.gif

上图显示了按照下述方式工作的流水线的组织形式。A型机器从输入库接受工件,对其施加操作A,得到的中间产品存放在缓冲库。B型机器从缓冲库接受中间产品,对其施加操作B,得到的最终产品存放在输出库。所有的机器平行并且独立地工作,每个库的容量没有限制。每台机器的工作效率可能不同,一台机器完成一次操作需要一定的时间。

给出每台机器完成一次操作的时间,计算完成A操作的时间总和的最小值,和完成B操作的时间总和的最小值。

格式

PROGRAM NAME: job

INPUT FORMAT:

(file job.in)

第一行 三个用空格分开的整数:N,工件数量 (1<=N<=1000);M1,A型机器的数量 (1<=M1<=30);M2,B型机器的数量 (1<=M2<=30)。

第二行…等 M1个整数(表示A型机器完成一次操作的时间,1..20),接着是M2个整数(B型机器完成一次操作的时间,1..20)

OUTPUT FORMAT:

(file job.out)

只有一行。输出两个整数:完成所有A操作的时间总和的最小值,和完成所有B操作的时间总和的最小值(A操作必须在B操作之前完成)。

SAMPLE INPUT

5 2 3
1 1 3 1 4

SAMPLE OUTPUT

3 5
【分析】:
    这道题用二分答案+贪心.
    首先对于A工序的总耗时二分一个答案x,然后验证这个时间是否合不合理:既不多,又不少.那么在这一段时间内可作出的工件数为∑x/time[i],1≤i≤M1,对该值进行判断即可.比较难想的是B工序的总耗时,首先算出完成一个工件的A工序后的时间点time1,如样例中A机器1 1,那么在时间ans1=3内这两台机器完成一个工件的时间分别为1,1,2,2,3,3,即表示:1时间点出1个工件<由A1机器制造>,1时间点出1个工件<由A2机器制造>,2时间点出1个工件<由A1机器制造>,2时间点出1个工件<由A2机器制造>,3时间点出1个工件<由A1机器制造>,3时间点出1个工件<由A2机器制造>.                              
    然后开始B工序的二分答案,如第一问二分方法,我们需检查二分出的数x是否满足条件.相应的,我们处理出所有完成一个工件的B工序后的时间点time2[i],那么x-time2[i]就是对于当前时间而言最晚开始B工序的时间点.类似的,我们将time2排序,筛出N个最小的time2,这样使得整体开始B工序的时间尽可能晚<若这里改为筛出N个最大的,那么这个二分永远找不出解,仔细想想>.
    最后判断是否对应的每个time1都小于等于time2,若不是就说明这种方案不行,应当继续二分
 
【代码】:
/*
ID:csyzcyj1
PROG:job
LANG:C++
*/
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<algorithm>
#include<iostream>
#include<vector>
#include<stack>
#include<queue>
using namespace std;
#define MAXM 31
#define MAX 20001
#define IMAX 2147483647
int N,M1,M2,ans1=IMAX,ans2=IMAX,min1=IMAX,min2=IMAX,pos1[MAX],pos2[MAX],tot=0;
int a[MAXM],b[MAXM];
bool cmp(int x,int y){return x>y;}
bool check1(int x)
{
int now=0;
for(int i=1;i<=M1;i++)
now+=x/a[i];
if(now>=N)   return true;
else return false;
}
bool check2(int x)
{
int sum=0;
memset(pos2,0,sizeof(pos2));
for(int i=1;i<=M2;i++)
for(int j=1;j<=x/b[i];j++)
if(x-j*b[i]>0)   
pos2[++sum]=j*b[i];//算出完成一个工件的B工序后的时间点
if(sum<N)   return false;
sort(pos2+1,pos2+1+sum);
for(int i=1;i<=N;i++)
if(pos1[i]>x-pos2[i])//x-pos2[i]表示最晚开始B工序的时间 
return false;
return true;         
}
void work()
{
int left=1,right=min1*N;
while(left<=right)//二分完成A操作的总耗时 
{
int middle=(left+right)/2;
if(check1(middle))
right=middle-1;
else  left=middle+1;
}
ans1=left;
for(int i=1;i<=M1;i++)
for(int j=1;j<=ans1/a[i];j++)
pos1[++tot]=j*a[i];//算出完成一个工件的A工序后的时间点
sort(pos1+1,pos1+1+tot,cmp);
for(int i=1;i<=N;i++)
pos1[i]=pos1[i+tot-N];//筛出最小的N个时间 
left=ans1,right=ans1+min2*N;//二分完成B操作的总耗时 
while(left<=right)
{
int middle=(left+right)/2;
if(check2(middle))
right=middle-1;
else  left=middle+1;
}
ans2=left;
}
int main()
{
freopen("job.in","r",stdin);
freopen("job.out","w",stdout); 
scanf("%d%d%d",&N,&M1,&M2);
for(int i=1;i<=M1;i++)
{
scanf("%d",&a[i]);
min1=min(min1,a[i]);
}
for(int i=1;i<=M2;i++)
{
scanf("%d",&b[i]);
min2=min(min2,b[i]);
}
work();
printf("%d %d\n",ans1,ans2);
//system("pause");
return 0;
}
【评测信息】:
      Compiling...
      Compile: OK
      Executing...
      Test 1: TEST OK [0.000 secs, 3536 KB]
      Test 2: TEST OK [0.000 secs, 3536 KB]
      Test 3: TEST OK [0.000 secs, 3536 KB]
      Test 4: TEST OK [0.000 secs, 3536 KB]
      Test 5: TEST OK [0.000 secs, 3536 KB]
      Test 6: TEST OK [0.000 secs, 3536 KB]
      Test 7: TEST OK [0.000 secs, 3536 KB]
      Test 8: TEST OK [0.000 secs, 3536 KB] 
      Test 9: TEST OK [0.000 secs, 3536 KB]
      Test 10: TEST OK [0.011 secs, 3536 KB]
      Test 11: TEST OK [0.011 secs, 3536 KB]
      Test 12: TEST OK [0.011 secs, 3536 KB]
      All tests OK.
 

转载注明出处:http://blog.csdn.net/u011400953

这篇关于USACO Training 4.2.3 Job Processing 工序安排 题解与分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1038501

相关文章

Nginx分布式部署流程分析

《Nginx分布式部署流程分析》文章介绍Nginx在分布式部署中的反向代理和负载均衡作用,用于分发请求、减轻服务器压力及解决session共享问题,涵盖配置方法、策略及Java项目应用,并提及分布式事... 目录分布式部署NginxJava中的代理代理分为正向代理和反向代理正向代理反向代理Nginx应用场景

Redis中的有序集合zset从使用到原理分析

《Redis中的有序集合zset从使用到原理分析》Redis有序集合(zset)是字符串与分值的有序映射,通过跳跃表和哈希表结合实现高效有序性管理,适用于排行榜、延迟队列等场景,其时间复杂度低,内存占... 目录开篇:排行榜背后的秘密一、zset的基本使用1.1 常用命令1.2 Java客户端示例二、zse

Redis中的AOF原理及分析

《Redis中的AOF原理及分析》Redis的AOF通过记录所有写操作命令实现持久化,支持always/everysec/no三种同步策略,重写机制优化文件体积,与RDB结合可平衡数据安全与恢复效率... 目录开篇:从日记本到AOF一、AOF的基本执行流程1. 命令执行与记录2. AOF重写机制二、AOF的

MyBatis Plus大数据量查询慢原因分析及解决

《MyBatisPlus大数据量查询慢原因分析及解决》大数据量查询慢常因全表扫描、分页不当、索引缺失、内存占用高及ORM开销,优化措施包括分页查询、流式读取、SQL优化、批处理、多数据源、结果集二次... 目录大数据量查询慢的常见原因优化方案高级方案配置调优监控与诊断总结大数据量查询慢的常见原因MyBAT

分析 Java Stream 的 peek使用实践与副作用处理方案

《分析JavaStream的peek使用实践与副作用处理方案》StreamAPI的peek操作是中间操作,用于观察元素但不终止流,其副作用风险包括线程安全、顺序混乱及性能问题,合理使用场景有限... 目录一、peek 操作的本质:有状态的中间操作二、副作用的定义与风险场景1. 并行流下的线程安全问题2. 顺

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

Java中最全最基础的IO流概述和简介案例分析

《Java中最全最基础的IO流概述和简介案例分析》JavaIO流用于程序与外部设备的数据交互,分为字节流(InputStream/OutputStream)和字符流(Reader/Writer),处理... 目录IO流简介IO是什么应用场景IO流的分类流的超类类型字节文件流应用简介核心API文件输出流应用文

SpringBoot集成XXL-JOB实现任务管理全流程

《SpringBoot集成XXL-JOB实现任务管理全流程》XXL-JOB是一款轻量级分布式任务调度平台,功能丰富、界面简洁、易于扩展,本文介绍如何通过SpringBoot项目,使用RestTempl... 目录一、前言二、项目结构简述三、Maven 依赖四、Controller 代码详解五、Service

解决Nginx启动报错Job for nginx.service failed because the control process exited with error code问题

《解决Nginx启动报错Jobfornginx.servicefailedbecausethecontrolprocessexitedwitherrorcode问题》Nginx启... 目录一、报错如下二、解决原因三、解决方式总结一、报错如下Job for nginx.service failed bec

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按