鸡蛋掉落

解析谷歌经典面试题——超级鸡蛋掉落问题。通过动态规划与二分查找结合的方式优化算法,降低时间复杂度。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

谷歌的一道经典面试题,题目如下:

https://leetcode-cn.com/problems/super-egg-drop/

给你 k 枚相同的鸡蛋,并可以使用一栋从第 1 层到第 n 层共有 n 层楼的建筑。

已知存在楼层 f ,满足 0 <= f <= n ,任何从 高于 f 的楼层落下的鸡蛋都会碎,从 f 楼层或比它低的楼层落下的鸡蛋都不会破。

每次操作,你可以取一枚没有碎的鸡蛋并把它从任一楼层 x 扔下(满足 1 <= x <= n)。如果鸡蛋碎了,你就不能再次使用它。如果某枚鸡蛋扔下后没有摔碎,则可以在之后的操作中 重复使用 这枚鸡蛋。

请你计算并返回要确定 f 确切的值 的 最小操作次数 是多少?

 
示例 1:

输入:k = 1, n = 2
输出:2
解释:
鸡蛋从 1 楼掉落。如果它碎了,肯定能得出 f = 0 。 
否则,鸡蛋从 2 楼掉落。如果它碎了,肯定能得出 f = 1 。 
如果它没碎,那么肯定能得出 f = 2 。 
因此,在最坏的情况下我们需要移动 2 次以确定 f 是多少。 
示例 2:

输入:k = 2, n = 6
输出:3
示例 3:

输入:k = 3, n = 14
输出:4
 

提示:

1 <= k <= 100
1 <= n <= 104

当把一枚鸡蛋从x楼层扔下时,那么有两种可能,鸡蛋碎了,或者没碎

(1)鸡蛋没碎,那么操作次数为1+f(k,n-x) 

(2)鸡蛋碎了,那么操作次数为1+f(k-1,n-1)

因为在最坏的情况下,我们也要知道确切值,那么此时最小次数应该为 1+ max(f(k,n-x),f(k-1,x-1))。

而我们要得到全部尝试的最小次数,则是找到一个可以使得如上值最小的x。即 1+ min( max(f(k,n-x),f(k-1,x-1)) )

如果我们去遍历每一层楼,来求得这个最小值,那么复杂度为kn^2。这种方法在leetcode上会超出时间限制。

 

优化后的方法如下:

二分法+动态规划

首先在k不变的前提下:

(1)f(k-1,x-1)是随着x单调递增的,当x增大,f(k-1,x-1)也会增大

(2)f(k,n-x)是随着x单调递减的,当x增大,f(k,n-x)会减小

参考LeetCode这张图:

fig1

所以我们用二分法来寻找x的值,复杂度为kn log(n)。

实现代码如下:

func superEggDrop(k int, n int) int {
	dp := make([][]int, k+1)
	for i := 1; i <= k; i++ {
		dp[i] = make([]int, n+1)
		dp[i][1] = 1
	}
	for i := 0; i <= n; i++ {
		dp[1][i] = i
	}

	r := calculate(k, n, dp)
	return r
}

func calculate(k, n int, dp [][]int) int {
	if n == 0 {
		return 0
	}

    // 如果已经存储,就不需要再计算
	if dp[k][n] != 0 {
		return dp[k][n]
	}

	low := 1
	high := n

	for low+1 < high {
		x := (low + high) / 2
		t1 := calculate(k-1, x-1, dp)
		t2 := calculate(k, n-x, dp)

		if t1 < t2 {
			low = x
		} else if t1 > t2 {
			high = x
		} else {
			low = x
			high = x
		}
	}

	lAns := max(calculate(k, n-low, dp), calculate(k-1, low-1, dp))
	hAns := max(calculate(k, n-high, dp), calculate(k-1, high-1, dp))
	ans := 1 + min(lAns, hAns)
	dp[k][n] = ans
	return ans
}
func max(a, b int) int {
	if a > b {
		return a
	}
	return b
}

func min(a, b int) int {
	if a < b {
		return a
	}
	return b
}

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

IreneByron

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值