题解 [HNOI2001]产品加工

题目描述

某加工厂有A、B两台机器,来加工的产品可以由其中任何一台机器完成,或者两台机器共同完成。由于受到机器性能和产品特性的限制,不同的机器加工同一产品所需的时间会不同,若同时由两台机器共同进行加工,所完成任务又会不同。某一天,加工厂接到n个产品加工的任务,每个任务的工作量不尽一样。
你的任务就是:已知每个任务在A机器上加工所需的时间t1, B机器上加工所需的时间t2及由两台机器共同加工所需的时间t3,请你合理安排任务的调度顺序,使完成所有n个任务的总时间最少。

输入输出格式

输入格式
(输入文件共n+1行)
第1行为 n。 n是任务总数(1≤n≤6000)
第i+1行为3个[0,5]之间的非负整数t1,t2,t3,分别表示第i个任务在A机器上加工、B机器上加工、两台机器共同加工所需要的时间。如果所给的时间t1或t2为0表示任务不能在该台机器上加工,如果t3为0表示任务不能同时由两台机器加工。

输出格式
最少完成时间

输入输出样例

输入样例#1

5
2 1 0
0 5 0
2 4 1
0 0 3
2 1 1

输出样例#1

9

解题思路

终于有一道状态转移方程面积不那么庞大的DP题了(手动滑稽_ hua|ji_)
很显然,如果我们分别考虑某个任务由第一个机器或者由第二个机器来完成的话,不管是时间复杂度还是空间复杂度都输无法承受的,那么我们来考虑压维
我们定义状态变量 f [ i ] 表示第一个机器工作了 i 分钟时第二个机器的工作时间,外层循环就是从 1 到 n ,也就是循环任务数量,当我们每枚举到一个任务时,先考虑它是否能够被安排给第二个机器,如果可以,我们就把 f [ i ] 加上当前的 t2 ,如果不行,就还把他设为一个极大值;然后我们再来考虑把它分给两个机器或是两个机器共同完成的最优解,如果可以被分配给第一个机器时,第二个机器的时间就不用增加,转移方程即为:

f [ i ] = min ( f [ i ] , f [ i - t1 ] ) ;

如果他能被分配给两个机器共同完成时,不仅第一个机器的时间要转移,第二个机器的时间也要增加,故转移方程为:

f [ i ] = min ( f [ i ] , f [ i - t3 ] + t3 ) ;

需要注意的细节是,每次进行转移的时候内层循环应循环完成任务的总时间,又因为我们压掉了循环物品的那一维,所以内层循环应从总时间循环到零,在这里我们可以进行一个优化,就是在读入的时候边读边进行状态转移,定义一个 sum 变量每次加上读入的 t1 , t2 , t3 中的最大值,循环结束时的 sum 即为完成任务的总时间
然后是最终答案,我们定义一个 ans ,初始化成极大值,然后循环总时间,每次取 min ( ans , max ( i , f [ i ] ) ) 即为最终答案
至于为什么是 max ( i , f [ i ] ) ,因为 i 表示第一个机器的时间,f [ i ]为第二个机器的时间,如果你只取较小的那个时间,则另一个机器很显然还没有完成任务,就不是满足条件的答案,只有你取了两个时间中的最大值才能保证两个机器都完成了任务
下面是C++代码,具体细节看注释

C++代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define rr register
using namespace std;

const int maxn=3e4+10;
const int inf=2147482647;
int n,t1,t2,t3,sum=0,ans=inf;
int f[maxn];

inline int read(){//珂朵莉版快读~~~
    int chtholly=0,willem=1;char c=getchar();
    while(c<'0' || c>'9'){if(c=='-')willem=-1;c=getchar();}
    while(c<='9' && c>='0'){chtholly=chtholly*10+(int)(c-'0');c=getchar();}
    return chtholly*willem;
}

int main(){
    memset(f,127/3,sizeof(f));
    f[0]=0;//初始化f
    n=read();for(rr int i=1;i<=n;i++){
        t1=read(),t2=read(),t3=read();
        sum+=max(t1,max(t2,t3));
        for(rr int j=sum;j>=0;j--){
            if(t2)f[j]+=t2;else f[j]=inf;
            if(t1 && j>=t1)f[j]=min(f[j],f[j-t1]);
            if(t3 && j>=t3)f[j]=min(f[j],f[j-t3]+t3);
        }
    }
    for(rr int i=0;i<=sum;i++)ans=min(ans,max(i,f[i]));
    printf("%d\n",ans);
    return 0;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容