LeetCode - 875. Koko Eating Bananas

Koko loves to eat bananas. There are N piles of bananas, the i-th pile has piles[i] bananas. The guards have gone and will come back in H hours.

Koko can decide her bananas-per-hour eating speed of K. Each hour, she chooses some pile of bananas, and eats K bananas from that pile. If the pile has less than K bananas, she eats all of them instead, and won't eat any more bananas during this hour.

Koko likes to eat slowly, but still wants to finish eating all the bananas before the guards come back.

Return the minimum integer K such that she can eat all the bananas within H hours.

Example 1:
Input: piles = [3,6,7,11], H = 8
Output: 4

Example 2:
Input: piles = [30,11,23,4,20], H = 5
Output: 30

Example 3:
Input: piles = [30,11,23,4,20], H = 6
Output: 23

Note:
1 <= piles.length <= 10^4
piles.length <= H <= 10^9
1 <= piles[i] <= 10^9

题意:

Koko是一只喜欢吃香蕉的大猩猩。现在有N堆香蕉,第i堆有piles[i]个香蕉。管理员现在离开了,会在H(H >= piles.length)小时之后回来。Koko可以决定吃香蕉的速度,即每小时吃K个香蕉。每个小时它选择一堆香蕉,并吃掉其中的K个香蕉。如果某一堆里剩下的香蕉少于K个,它会吃完这一堆剩下的所有香蕉,但它这一小时内不会吃更多的其他堆的香蕉。

假设Koko喜欢慢慢地吃香蕉,但又想在管理员回来之前吃完所有香蕉。请问Koko每小时至少需要吃多少香蕉才能在管理员回来之前吃完呢?

例如,有4堆香蕉,每堆香蕉的数目为[3, 6, 7,11],如果管理员在8小时后回来,那么Koko每小时吃4个香蕉。

分析:

由于Koko在一个小时内把一堆香蕉吃完之后不会再去吃其他的香蕉,那么它一小时能吃掉的香蕉的数目不会超过最多的一堆香蕉的数目(记为M)。同时,它每小时最少会吃1个香蕉,所以最终Koko决定的吃香蕉的速度K应该是在1到M之间。

我们可以应用二分查找的思路,先选取1和M的平均数,(1+M)/2,看以这个速度Koko能否在H小时内吃掉所有香蕉。如果不能在H小时内吃掉所有的香蕉,那么它需要尝试更快的速度,也就是K应该在(1+M)/2到M之间,下一次我们尝试(1+M)/2和M的平均值。

如果Koko以(1+M)/2的速度能够在H小时内吃完所有的香蕉,那么我们来判断这是不是最慢的速度。可以尝试一下稍微慢一点的速度,(1+m)/2 - 1。如果Koko以这个速度不能在H小时之内吃完所有香蕉,那么(1+M)/2就是最慢的可以在H小时吃完香蕉的速度。如果以(1+m)/2 - 1的速度也能在H小时内吃完香蕉,那么接下来Koko尝试更慢的速度,1和(1+M)/2的平均值。

以此类推,我们按照二分查找的思路总能找到让Koko在H小时内吃完所有香蕉的最慢速度K。以下是这种思路的参考代码:

	public static int method(int[] piles, int H) {
		int min = 1;
		int max = 1;
		for (int pile: piles) {
			if (pile > max) {
				max = pile;
			}
		}
		while (min < max) {
			int mid = min + (max - min) / 2;
			int hours = getHours(piles, mid);
			if (hours <= H) {
				if (mid == min || getHours(piles, mid-1) > H) {
					return mid;
				}
				max = mid - 1;
			} else {
				min = mid + 1;
			}
		}
		return -1;   // 无解
	}
	
	/**
	 * 以speed的速度吃香蕉,吃完所有的香蕉需要多少小时
	 * @param piles
	 * @param speed
	 * @return
	 */
	public static int getHours(int[] piles, int speed) {
		int hours = 0;
		for (int pile : piles) {
			int hour = (pile + speed - 1) / speed;
			hours += hour;
		}
		return hours;
	}

时间空间复杂度:

在上述代码中,getHours需要扫描整个数组,每执行一次需要O(n)的时间。函数getHours执行的次数等于minEatingSpeed中while循环执行的次数。由于应用了二分查找算法,while循环执行的次数为O(logMAX),max为最多一堆香蕉的数目。因此总的时间复杂度是O(nlogMAX)。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值