28.考试

目录

题目

Description

Input

Output

Note

思路

C++完整代码(含注释)


题目

Description


小学期马上就要结束了,为了检验大家的学习成果,老师进行了一次考试。然而小徐前两周半都忙于练习篮球,几乎没有学习,因此考试时很可能做不完所有题目。

但小徐仍然想要拿到尽可能高的分数,因此在做题时需要合理做出取舍。已知考试时间为 

n

 分钟,小徐希望在不超时的情况下,考出最高的分数。

设做出第

j

道试题用时为

t[j]

, 该试题的”性价比”为

w[j]

, ( 

t[j] * w[j]

为做出该题的得分 )

若选择了

k

题进行作答,则所求的分数之和为

t[j_{1}] \times w[j_{1}] \quad + \quad t[j_{2}] \times w[j_{2}] \quad + \quad ... \quad + \quad t[j_{k}] \times w[j_{k}]



请你帮小徐算出他最高能得多少分。

Input

第一行,为2个正整数,用一个空格隔开: 

n

,

m

,其中

n

表示总分钟数, 

m

为考试的题目数量

从第2行到第

m+1

行,第

j

行给出了编号为

j-1

的题目的基本数据,每行有2个非负整数

t

,

w

 .其中

t

表示小徐做出该题目的所用时间,

w

表示该题目的”性价比”。

Output

一个整数,表示小徐最高能得多少分。

Note

数据保证

n \lt 30000

m \lt 25

, 

v\leq10000

, 

w

1-5

测试输入期待的输出时间限制内存限制额外进程
测试用例 1以文本方式显示
  1. 1000 5↵
  2. 800 2↵
  3. 400 5↵
  4. 300 5↵
  5. 400 3↵
  6. 200 2↵
以文本方式显示
  1. 3900↵
1秒64M0
测试用例 2以文本方式显示
  1. 22415 2↵
  2. 2824 3↵
  3. 2485 3↵
以文本方式显示
  1. 15927↵
1秒64M0


思路

本题是动态规划中的01背包问题.

第一步:dp数组以及下标的含义

我们定义的dp数组是一个一维数组,长度为n+1;

dp[i]表示总时间为i时获得的最高分数;

换言之,dp[i]存储了用总时间为i所能获得的最高分数。

第二步:确定递推公式

递推公式为dp[j] = max(dp[j], dp[j - que[i][0]] + sco[i])

表示要获得总时间为j的最高分数,可以选择当前题目或不选择当前题目。通过比较两种选择的得分,取得分较大的那个作为dp[j]的值。

第三步:dp数组初始化

dp数组全部初始化为0。在动态规划过程中,由于dp数组表示的是最大分数,因此初始化为0是合适的。

第四步:确定遍历顺序

在代码中,遍历的顺序是先遍历题目,再遍历总时间。

外层循环控制题目的遍历,内层循环控制总时间的遍历。

这样的遍历顺序可以确保每个题目只选一次,并且保证了在计算dp[j]时,dp[j - que[i][0]]已经被更新过(即内层循环从大到小的顺序遍历)。通过这样的遍历顺序,我们可以逐步计算出dp数组中每个位置的值,从而得到最终的最高分数。


C++完整代码(含注释)

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

int main() {
    int n, m;
    cin >> n >> m;
    vector<vector<int>> que(m, vector<int>(2));  // 存储题目的时间和性价比
    vector<int> sco(m);  // 存储题目的得分
    for (int i = 0; i < m; i++) {
        cin >> que[i][0] >> que[i][1];  // 读入每个题目的时间和性价比
        sco[i] = que[i][0] * que[i][1];  // 计算每个题目的得分
    }
    vector<int> dp(n + 1, 0);  // 动态规划数组,dp[i]表示总时间为i时获得的最高分数
    for (int i = 0; i < m; i++) {
        for (int j = n; j >= que[i][0]; j--) {
            dp[j] = max(dp[j], dp[j - que[i][0]] + sco[i]);  // 状态转移方程,更新dp[j]为包括或不包括当前题目得分的最大值
        }
    }
    cout << dp[n] << endl;  // 输出最终总时间为n时获得的最高分数
    return 0;
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

榆榆欸

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值