leetcode
leetcode 1051 ~ 1100
填充书架

填充书架

难度:

标签:

题目描述

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

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

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

需要注意的是,在上述过程的每个步骤中,摆放书的顺序与给定图书数组 books 顺序相同

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

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

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

 

示例 1:

输入: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

代码结果

运行时间: 23 ms, 内存: 16.2 MB


/*
 * 思路:
 * 1. 使用动态规划和Java Stream来解决这个问题。
 * 2. dp数组记录每本书的最小高度。
 * 3. 使用Stream对数组进行处理,找到每种可能的最小高度。
 */

import java.util.stream.IntStream;

public int minHeightShelves(int[][] books, int shelfWidth) {
    int n = books.length;
    int[] dp = new int[n + 1];
    Arrays.fill(dp, Integer.MAX_VALUE);
    dp[0] = 0;

    IntStream.rangeClosed(1, n).forEach(i -> {
        int[] widthHeight = {0, 0};
        IntStream.iterate(i, j -> j > 0, j -> j - 1).forEach(j -> {
            widthHeight[0] += books[j - 1][0];
            if (widthHeight[0] > shelfWidth) return;
            widthHeight[1] = Math.max(widthHeight[1], books[j - 1][1]);
            dp[i] = Math.min(dp[i], dp[j - 1] + widthHeight[1]);
        });
    });
    return dp[n];
}

解释

方法:

此题解使用动态规划解决问题。定义dp数组f,其中f[i]表示放置前i本书的最小总高度。初始化f[0]为0,表示没有书时高度为0。对于每本书i,从i开始向前检查以当前书为结束的所有可能的书层配置。对每种配置,计算该层的最大高度和当前层的宽度总和。如果宽度总和没有超过书架宽度,更新f[i]为当前最优解。这样,最终f[n]中存储的就是放置所有书的最小高度。

时间复杂度:

O(n^2)

空间复杂度:

O(n)

代码细节讲解

🦆
在你的动态规划解决方案中,如何处理书的顺序?是否考虑了题目要求的‘按顺序’摆放书籍的规则?
在我的动态规划解决方案中,通过从前向后遍历书籍列表并逐一决定每本书的放置,来保证书籍是按照给定的顺序摆放的。内部循环从当前书籍索引i开始,并向前检查,这样可以确保书籍的顺序从1到i是连续的,符合题目要求的'按顺序'摆放规则。
🦆
动态规划数组f[i]的初始化为什么选择float('inf'),除了f[0]外,这样的初始化对算法的影响是什么?
在动态规划中,f[i]数组的初始化为float('inf')(意味着无穷大)用于确保在开始算法时,任何非初始状态(即f[0]除外)都不会被错误地认为是一个可行的最小值。这种初始化方式允许我们在更新每个f[i]时,只考虑那些实际上通过比较得到的更小的值。这样可以保证,每次更新f[i]都是基于有效的、可能的最小高度,而不是一个未初始化或错误初始化的值。
🦆
在内层循环中,当书的宽度累加超过书架宽度时就会停止循环。这是否意味着我们总是在宽度超出前就不能再添加新的书?这种方式是否最优?
是的,内层循环中当书的宽度累加超过书架宽度时停止循环,确保不会添加超过书架宽度的书籍。这种设计是为了遵守书架宽度的限制,并尝试找出在不超过宽度限制的情况下的可能的最佳高度配置。虽然这种方法可能不会在每次都找到全局最优解,但它是一个有效的启发式方法,通常可以在合理的时间内找到非常接近最优的解决方案。
🦆
你的算法中提到更新最小高度时使用了条件判断`if f[j - 1] + mx < f[i]`。这里的逻辑是否保证了每次都能获得局部最优解,进而实现全局最优解?
这个条件判断确保我们在每个步骤中都寻求局部最优解,即在给定当前书的配置下,尝试找到最小的可能高度。通过动态规划的特性,这种局部最优的决策能够累积起来,努力朝向全局最优解。虽然动态规划通常能提供全局最优解,但这也取决于问题的具体构造和状态转移的定义。在大多数情况下,本算法能有效地找到整体的最小高度。

相关问题