1. 问题描述:
传送带上的包裹必须在 D 天内从一个港口运送到另一个港口。
传送带上的第 i 个包裹的重量为 weights[i]。每一天,我们都会按给出重量的顺序往传送带上装载包裹。我们装载的重量不会超过船的最大运载重量。
返回能在 D 天内将传送带上的所有包裹送达的船的最低运载能力
示例 1:
输入:weights = [1,2,3,4,5,6,7,8,9,10], D = 5
输出:15
解释:
船舶最低载重 15 就能够在 5 天内送达所有包裹,如下所示:
第 1 天:1, 2, 3, 4, 5
第 2 天:6, 7
第 3 天:8
第 4 天:9
第 5 天:10
请注意,货物必须按照给定的顺序装运,因此使用载重能力为 14 的船舶并将包装分成 (2, 3, 4, 5), (1, 6, 7), (8), (9), (10) 是不允许的
示例 2:
输入:weights = [3,2,2,4,1,4], D = 3
输出:6
解释:
船舶最低载重 6 就能够在 3 天内送达所有包裹,如下所示:
第 1 天:3, 2
第 2 天:2, 4
第 3 天:1, 4
示例 3:
输入:weights = [1,2,3,1,1], D = 4
输出:3
解释:
第 1 天:1
第 2 天:2
第 3 天:3
第 4 天:1, 1
提示:
1 <= D <= weights.length <= 50000
1 <= weights[i] <= 500
2. 思路分析:
① 一看题目感觉毫无头绪,感觉处理起来比较麻烦,但是仔细分析一看这个其实是一个二分查找的模型,假如你能够清楚使用二分查找那么剩下来的就比较好办了,能够使用二分查找的题目有明显的特点,问题主要是分为两类:"最小化最大值问题"和"最大化最小值问题",我们在做题目的时候需要根据题目的字眼想到相关的模型之类的,这道题目有一个比较明显限制那就是在D天之内需要需要运输完所有的包裹,所以这里包含一个最大的范围,所以属于一个最大值的问题,第二个从题目中求解的最低运载能力可以知道需要最小化载货的重量,所以第二个最小化也找到了,属于最小化最大值问题,对于这种问题使用二分查找就最合适了
② 确定了使用二分查找的方法之后,接下来需要确定二分查找的几个处理细节,主要分为以下几个步骤:
(1) 需要确定二分查找的范围,左边界l与右边界r,我们只能够在这个范围内找到符合题目要求的答案,由题目可以知道最小的载货量为所有包裹中的最大值,也就是max(weights):因为当载货量大于了包裹的重量的时候才可以运输当前的货品,最大的载货量肯定为所有货品的重量之和,这样我们在一天就可以将其运输完,所以左边界与右边界的范围就确定了
(2) 需要确定更新左右边界的策略,根据题目可以知道我们要求解的是D天之内需要将所有的货运输完,所以当我们求解出mid = (l + r) / 2的时候假设当前的载货量mid能够在D天之内完成,这个时候因为要求解最小载货量,所以肯定是右边界往左边的,这样才可以尝试更小的载货量在D天之内完成,假如不能够在D天之内完成这个时候就将左边界向右扩展,说明需要载货量更大才可以在D天之内完成
(3) 检查D天之内是否能够运输完的方法也很简单,累加当前的载货量,假如累加载货量大于了最大的载货量说明累加天数即可,需要注意一个细节是因为最后一个货品的时候有可能也是满足最大载货量的,这个时候需要累加多一天,这里可以使用一个技巧就是在初始化天数的时候为1天,这样就可以避免最后一天没有被累加上去的问题
(4) 需要确定二分查找的返回的结果,从题目中可以知道我们是尽量往载货量小的方向进行缩小的,假设载货量最小,那么r会一直减少,直到小于了l,所以这个时候应该是返回l的
③ 遇到这种最大最小字眼问题的时候需要想到二分查找的模型,假如题目满足最大化最小值或者最小化最大值问题都是可以在一个范围内通过二分查找解决的
3. 代码如下:
from typing import List
class Solution:
"""
:param minweight: 表示当前的最低载货量
"""
def check(self, weights: List[int], D: int, minweight: int):
# 这里可以初始化天数为1
cur, days = 0, 1
for i in weights:
if cur + i <= minweight:
cur += i
else:
cur = i
days += 1
return days <= D
# 二分的思路是在D天的限制条件下看是否能够运输完来缩小与扩展左右边界
def shipWithinDays(self, weights: List[int], D: int) -> int:解决
l, r = 0, 0
for i in weights:
l = max(l, i)
r += i
while l <= r:
# 在D天能够将物品运输完这个时候就需要右边界减小
mid = (l + r) // 2
if self.check(weights, D, mid):
r = mid - 1
# 否则说明在D天之内载货量mid不能够全部运输完
else:
l = mid + 1
return l