LC-1105. 填充书架(记忆化搜索 ==> 动态规划)

文章介绍了如何使用记忆化搜索和递推方法解决LeetCode中的一个题目,即在给定书的厚度、高度和书架宽度的情况下,如何摆放书以使书架的整体高度最小。通过示例和代码解释了如何进行状态转移并避免重复计算,以优化解决方案的效率。
摘要由CSDN通过智能技术生成

1105. 填充书架

难度中等169

给定一个数组 books ,其中 books[i] = [thicknessi, heighti] 表示第 i 本书的厚度和高度。你也会得到一个整数 shelfWidth

按顺序 将这些书摆放到总宽度为 shelfWidth 的书架上。

先选几本书放在书架上(它们的厚度之和小于等于书架的宽度 shelfWidth ),然后再建一层书架。重复这个过程,直到把所有的书都放在书架上。

需要注意的是,在上述过程的每个步骤中,摆放书的顺序与你整理好的顺序相同

  • 例如,如果这里有 5 本书,那么可能的一种摆放情况是:第一和第二本书放在第一层书架上,第三本书放在第二层书架上,第四和第五本书放在最后一层书架上。

每一层所摆放的书的最大高度就是这一层书架的层高,书架整体的高度为各层高之和。

以这种方式布置书架,返回书架整体可能的最小高度。

示例 1:

img

输入:books = [[1,1],[2,3],[2,3],[1,1],[1,1],[1,1],[1,2]], shelfWidth = 4
输出:6
解释:
3 层书架的高度和为 1 + 3 + 2 = 6 。
第 2 本书不必放在第一层书架上。

示例 2:

输入: books = [[1,3],[2,4],[3,2]], shelfWidth = 6
输出: 4

提示:

  • 1 <= books.length <= 1000
  • 1 <= thicknessi <= shelfWidth <= 1000
  • 1 <= heighti <= 1000

记忆化搜索 ==> 递推

题解:https://leetcode.cn/problems/filling-bookcase-shelves/solution/jiao-ni-yi-bu-bu-si-kao-dong-tai-gui-hua-0vg6/

记忆化

class Solution {
    int[][] books;
    int[] cache;
    int shelfWidth;
    public int minHeightShelves(int[][] books, int shelfWidth) {
        this.books = books;
        this.shelfWidth = shelfWidth;
        // 举个例子,[把 books[n -1] 单独放一层,把 books[n - 2] 单独放1层和[把 books[n - 2] 和 books[n -1]放同一层] ,都会递归到dfs(n - 3)。
        // 一叶知秋,整个递归中有大量重复递归调用(递归入参相同
        cache = new int[books.length]; 
        Arrays.fill(cache, -1);
        return dfs(books.length - 1);
    }

    /**
        定义dfs(i) 表示把books[0]到books[i]按顺序摆放后的最小高度
        枚举最后一层的第一本书的下标j,那么从books[j]到books[i]的宽度之和不能超过shelfWidth
        状态转移方程:dfs(i) = min{dfs(j-1) + max(books[k][1])}, j <= k= i
        递归边界:dfs(-1) = 0,没有书,最小高度为0
        递归入口:dfs(n-1) 表示把书按顺序拜访后的最小高度 
     */
    public int dfs(int i){
        if(i < 0) return 0; // 没有书了, 高度为0
        if(cache[i] >= 0) return cache[i];
        int res = Integer.MAX_VALUE, maxH = 0,leftW = shelfWidth;
        for(int j = i; j >= 0; j--){
            leftW -= books[j][0];
            if(leftW < 0) break; // 空间不足,无法放书
            maxH = Math.max(books[j][1], maxH); // 从j到i的最大高度
            res = Math.min(res, dfs(j-1) + maxH);
        }
        return cache[i] = res;
    }
}

转成递推

class Solution {
    public int minHeightShelves(int[][] books, int shelfWidth) {
        int n = books.length;
        int[] f = new int[n+1];// 由于需要一个状态标识i<0,因此整体右移,最后答案返回f[n]
        f[0] = 0; // 初始值:没有书时最小高度为0 
        for(int i = 0; i < n; i++){
            f[i+1] = Integer.MAX_VALUE; // 初始高度无穷大(res,因为求最小高度)
            int maxH = 0, leftW = shelfWidth;
            for(int j = i; j >= 0; j--){
                leftW -= books[j][0];
                if(leftW < 0) break; // 空间不足,无法放书
                maxH = Math.max(maxH, books[j][1]); // 从 j 到 i 的最大高度
                f[i+1] = Math.min(f[i+1], f[j] + maxH);
            }
        }
        return f[n];
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值