文章

10

粉丝

66

获赞

5

访问

46.2k

头像
题解:记忆化搜索(日期变量型)
P896 上海交通大学2021年机试题
发布于2022年5月22日 12:16
阅读数 4.8k

思路和算法

我们用 dp(i) 来表示从第 i 天开始到一年的结束,我们需要花的钱。考虑到一张通行证可以让我们在「接下来」的若干天进行旅行,所以我们「从后往前」倒着进行动态规划。

对于一年中的任意一天:

如果这一天不是必须出行的日期,那我们可以贪心地选择不买。这是因为如果今天不用出行,那么也不必购买通行证,并且通行证越晚买越好。所以有 dp(i)=dp(i+1);

如果这一天是必须出行的日期,我们可以选择买 1,7 或 30 天的通行证。若我们购买了 j 天的通行证,那么接下来的 j - 1 天,我们都不再需要购买通行证,只需要考虑第 i + j天及以后即可。因此,我们有
dp(i)=min{cost(j)+dp(i+j)},j∈{1,7,30}

其中 cost(j) 表示 j 天通行证的价格。为什么我们只需要考虑第 i+j 天及以后呢?这里和第一条的贪心思路是一样的,如果我们需要购买通行证,那么一定越晚买越好,在握着一张有效的通行证的时候购买其它的通行证显然是不划算的。

由于我们是倒着进行动态规划的,因此我们可以使用记忆化搜索,减少代码的编写难度。我们使用一个长度为 366的数组(因为天数是 [1, 365],而数组的下标是从 0 开始的)存储所有的动态规划结果,这样所有的 dp(i) 只会被计算一次(和普通的动态规划相同),时间复杂度不会增大。

最终的答案记为 dp(1)。

#include <bits/stdc++.h>
using namespace std;

class Solution {
    unordered_set<int> dayset;
    vector<int> costs;
    int memo[366] = {0};

public:
    int mincostTickets(vector<int>& days, vector<int>& costs) {
        this->costs = costs;
        for (int d: days) {
            dayset.insert...
登录查看完整内容


登录后发布评论

暂无评论,来抢沙发