ACM动态规划训练

题目描述

尼克每天上班之前都连接上英特网,接收他的上司发来的邮件,这些邮件包含了尼克主管的部门当天要完成的全部任务,每个任务由一个开始时刻与一个持续时间构成。尼克的一个工作日为N分钟,从第一分钟开始到第N分钟结束。当尼克到达单位后他就开始干活。如果在同一时刻有多个任务需要完成,尼克可以任选其中的一个来做,而其余的则由他的同事完成,反之如果只有一个任务,则该任务必需由尼克去完成,假如某些任务开始时刻尼克正在工作,则这些任务也由尼克的同事完成。如果某任务于第P分钟开始,持续时间为T分钟,则该任务将在第P+T-1分钟结束。写一个程序计算尼克应该如何选取任务,才能获得最大的空暇时间。

输入

输入数据第一行包含两个用空格隔开的整数N和K,1≤N≤10000,1≤K≤10000,N表示尼克的工作时间,单位为分,K表示任务总数。接下来共有K行,每一行有两个用空格隔开的整数P和T,表示该任务从第P分钟开始,持续时间为T分钟,其中1≤P≤N,1≤P+T-1≤N。

输出

输出文件仅一行包含一个整数表示尼克可能获得的最大空暇时间。

样例输入

15 6
1 2
1 6
4 11
8 5
8 1
11 5

样例输出

4



本题目需要从后向前推理,如果从前向后推理是后效性的,因为计算完前面的结果之后无法统计起来

【工作的结束时如果可以休息,后面就可以休息,直到下一个开始点】

【但如果从某开始点计算,不能够确定这次是休息还是工作



#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
using namespace std;
const int mxn=30000;
int n,k;
int p[mxn],a[mxn];
int f[mxn];
int main(){
    scanf("%d%d",&n,&k);
    int i,j;
    for(i=1;i<=k;i++){
        scanf("%d%d",&p[i],&a[i]);
    }
    for(i=n;i;i--){
        int flag=0;
        for(j=1;j<=k;j++){
            if(p[j]==i){
                flag=1;
                f[i]=max(f[i],f[i+a[j]]);
            }
        }
        if(!flag)f[i]=f[i+1]+1;
    }
    printf("%d\n",f[1]);
    return 0;
}

洛谷1417 烹调方案

本题地址: http://www.luogu.org/problem/show?pid=1417

题目背景

由于你的帮助,火星只遭受了最小的损失。但gw懒得重建家园了,就造了一艘飞船飞向遥远的earth星。不过飞船飞到一半,gw发现了一个很严重的问题:肚子饿了~ gw还是会做饭的,于是拿出了储藏的食物准备填饱肚子。gw希望能在T时间内做出最美味的食物,但是这些食物美味程度的计算方式比较奇葩,于是绝望的gw只好求助于你了。

题目描述

一共有n件食材,每件食材有三个属性,ai,bi和ci,如果在t时刻完成第i样食材则得到ai-t*bi的美味指数,用第i件食材做饭要花去ci的时间。
众所周知,gw的厨艺不怎么样,所以他需要你设计烹调方案使得美味指数最大

输入输出格式

输入格式:

第一行是两个正整数T和n,表示到达地球所需时间和食材个数。
下面一行n个整数,ai
下面一行n个整数,bi
下面一行n个整数,ci

输出格式:

输出最大美味指数

输入输出样例

输入样例#1:

74 1

502

2

47

输出样例#1:

408

说明

【数据范围】
对于40%的数据1<=n<=10
对于100%的数据1<=n<=50
所有数字均小于100,000

【思路】

   排序+背包。

   本题的特殊性在于其价值的计算,其代价是与放置食材的时间有关联的。如果没有tb的话则问题就成了一个01背包,即选与不选。

   首先根据如下规则对食材进行排序:

 

   排序的目的在于消除选择时间的影响。

   然后01背包。

   数据中有个比较。。的点过不去。

【代码】

 

 

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<algorithm>
 4 using namespace std;
 5 
 6 typedef long long LL;
 7 const int maxn = 100+10;
 8 struct Node{
 9     int a,b,c;
10     bool operator <(const Node& rhs) const{
11        return c*rhs.b<rhs.c*b;
12     }
13 }nodes[maxn];
14 LL d[100010];
15 int n,m;
16 
17 int main() 
18 {
19     scanf("%d%d",&m,&n);
20     for(int i=1;i<=n;i++) scanf("%d",&nodes[i].a);
21     for(int i=1;i<=n;i++) scanf("%d",&nodes[i].b);
22     for(int i=1;i<=n;i++) scanf("%d",&nodes[i].c);
23     
24     sort(nodes+1,nodes+1+n);
25     for(int i=1;i<=n;i++) if(nodes[i].c<=m)
26         for(int j=m;j>=nodes[i].c;j--)
27        {
28               d[j]=max(d[j],d[j-nodes[i].c]+nodes[i].a-j*nodes[i].b);
29        }
30     LL ans=0;
31     for(int i=0;i<=m;i++) ans=max(ans,d[i]);
32     printf("%lld",ans);
33     return 0;
34 }

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值