Egg Dropping Puzzle
The Two Egg Problem
曾经是Google的一道经典题。
题意:有一个百层高楼,鸡蛋在\(L\)层及以下扔都不碎,在\(L\)层以上都会碎。现在某人有\(k\)个鸡蛋,问在最坏情况下,至少扔多少次(用\(m\)表示)可以确定\(L\)的值。
分析:先来考虑\(k=1\)的情况。只有1个鸡蛋,为了得到一个确定的\(L\),只能从第一层开始,逐渐尝试增加楼层高度,因此\(m=100\)时,无论\(L\)的值是多少,都可以被确定。
再来考虑\(k=\infty\)的情况。这种情况就变为了binary search的问题,先拿一个在50层扔,如果碎,则在25层扔;如果不碎,则在75层扔...即\(m=7\)。
最后来考虑\(k=2\)的情况。因为\(k=1\)只能一层一层试,所以第一个鸡蛋应该尽可能缩小搜索空间,但是如果第一个鸡蛋的楼层间隔太小(比如在2层、4层...),无疑会增加\(m\)。不妨取第一个鸡蛋在10层、20层...,共10次;假如第一个在10层没碎,在20碎了,那么第二个鸡蛋可以尝试11、12...19,共9次;故\(m=19\)。
上面方案的问题在于:如果临界楼层比较高,那么第二个鸡蛋的次数是确定的,但第一个就需要多试几次,总次数就会增加。
那么如何使得不论临界楼层在哪,\(m\)的值都不会波动呢?
很简单,只要第一个多扔一次,确定的范围(第二个要试的次数)减小,总次数就会均衡。
对于第一个鸡蛋,第一次在\(a\)层扔,如果不碎,第二次向上增加\(a-1\)层...直到最后只向上增加\(1\)层:\(a+(a-1)+...+1\geq100\),故\(a\geq13.7\)。
鸡蛋一:在14层、27层、39层、50层、60层、69层、77层、84层、90层、95层、99层、100层扔,共12次;
鸡蛋二:如果蛋一在14层碎了,蛋二要扔13次,共14次;如果蛋一在27层碎了,蛋二要扔12次,共14次...故\(m=14\)。
Super Egg Problem
对于\(k=2\),我们有了一个比较好的解决方案。那么现在有\(k\)个鸡蛋,楼高\(n\)层,问题(记作\(m(k,n)\))又该如何解决?
\(k=1\)和\(n=1\)的情况比较简单:
| n\k | 1 | 2 | 3 | 4 | ... |
|---|---|---|---|---|---|
| 1 | 1 | 1 | 1 | 1 | |
| 2 | 2 | ||||
| 3 | 3 | ||||
| ... |
那么如果我们递归地思考:任选一层\(h\)扔第一个鸡蛋,无非有碎和不碎2种情况:
碎:临界楼层在1~h之间,问题规模缩小为\(m(k-1,h-1)\);
不碎:临界楼层在h~n之间,问题规模缩小为\(m(k,n-h)\)。
所以:\(m_h(k,n)=1+max\{ m(k-1,h-1),m(k,n-h)\}\)。
对于\(h\),可以采用枚举的方法,计算\(m_h(k,n)\),在其中选出一个最小的值,故问题得到解决:
\]
Base Case就是\(k=1\)和\(n=1\)。
时间复杂度\(O(KN^2)\),空间复杂度\(O(KN)\)。
记忆化递归:
class Solution {
public:
int superEggDrop(int K, int N) {
vector<vector<int>> memo(K + 1, vector<int>(N + 1, 0));
return helper(K, N, memo);
}
private:
int helper(int K, int N, vector<vector<int>>& memo) {
if(K == 1) {
return N;
}
if(N <= 1) {
return N;
}
if(memo[K][N]) {
return memo[K][N];
}
int ans = INT_MAX;
for(int i = 1;i <= N;++i) {
ans = min(ans, 1 + max(superEggDrop(K - 1, i - 1), superEggDrop(K, N - i)));
}
memo[K][N] = ans;
return ans;
}
};
上述最直观的解法复杂度太高,无法通过Leetcode的数据。如何优化呢?
\(m(K,N)\)表示该问题的解,如果鸡蛋数\(K\)固定,随着楼层数\(N\)增加,问题的解一定是增加的。
我们又把问题分解为了2个子问题\(m(k-1,h-1)\)和\(m(k,n-h)\),\(m(k-1,h-1)\)随\(h\)单调递增,\(m(k,n-h)\)单调递减(图源):

此时\(min(max(...))\)就是求中间的折点,可以使用Binary Search,时间复杂度降为\(O(KNlgN)\):
class Solution {
public:
int superEggDrop(int K, int N) {
vector<vector<int>> memo(K + 1, vector<int>(N + 1, 0));
return helper(K, N, memo);
}
private:
int helper(int K, int N, vector<vector<int>>& memo) {
if(K == 1) {
return N;
}
if(N <= 1) {
return N;
}
if(memo[K][N]) {
return memo[K][N];
}
int ans = INT_MAX;
int l = 1, r = N + 1;
while(l < r) {
int m = l + (r - l) / 2;
int broken = helper(K - 1, m - 1, memo), noBroken = helper(K, N - m, memo);
if(broken > noBroken) {
r = m;
ans = min(ans, 1 + broken);
}
else {
l = m + 1;
ans = min(ans, 1 + noBroken);
}
}
memo[K][N] = ans;
return ans;
}
};
当然,此题还有\(O(KN)\)的做法,甚至还有一种数学做法可以达到\(O(KlgN)\)的时间复杂度和\(O(1)\)的空间复杂度,由于本人水平实在有限,就不再探索。
Egg Dropping Puzzle的更多相关文章
- Egg Dropping Puzzle问题的分析
首先,基本问题是这样:You are given two eggs, and access to a 100-storey building. The aim is to find out the h ...
- 扔鸡蛋问题具体解释(Egg Dropping Puzzle)
经典的动态规划问题,题设是这种: 假设你有2颗鸡蛋,和一栋36层高的楼,如今你想知道在哪一层楼之下,鸡蛋不会被摔碎,应该怎样用最少的測试次数对于不论什么答案楼层都可以使问题得到解决. 假设你从某一层楼 ...
- 扔鸡蛋问题详解(Egg Dropping Puzzle)
http://blog.csdn.net/joylnwang/article/details/6769160 经典的动态规划问题,题设是这样的:如果你有2颗鸡蛋,和一栋36层高的楼,现在你想知道在哪一 ...
- 动态规划法(六)鸡蛋掉落问题(一)(egg dropping problem)
继续讲故事~~ 这天,丁丁正走在路上,欣赏着路边迷人的城市风景,突然发现前面的大楼前围了一波吃瓜群众.他好奇地凑上前去,想一探究竟,看看到底发生了什么事情. 原来本市的一位小有名气的科学家 ...
- 2 Egg Problem
继续我们的推理问题之旅,今天我们要对付的是一个Google的面试题:Two Egg Problem. 我们开始吧! No.2 Google Interview Puzzle : 2 Egg Prob ...
- 【LeetCode】887. Super Egg Drop 解题报告(Python)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 参考资料 日期 题目地址:https://leetc ...
- ERROR 1010 (HY000): Error dropping database (can't rmdir './test/', errno: 17)
在删除数据库的时候报标题所示错误 mysql> drop database test; ERROR (HY000): Error dropping database (can't rmdir ' ...
- Puzzle 面向服务/切面(AOP/IOC)开发框架 For .Net
Puzzle 面向服务/切面AOP开发框架 For .Net AOP主要实现的目的是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的隔离效 ...
- HDU5456 Matches Puzzle Game(DP)
题目 Source http://acm.hdu.edu.cn/showproblem.php?pid=5456 Description As an exciting puzzle game for ...
随机推荐
- MyBatis(十):Mybatis缓存的重要内容
本文是按照狂神说的教学视频学习的笔记,强力推荐,教学深入浅出一遍就懂!b站搜索狂神说或点击下面链接 https://space.bilibili.com/95256449?spm_id_from=33 ...
- css | js 实现扩展卡片小demo
1.代码如下 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UT ...
- ArrayList、LinkedList和Vector源码分析
ArrayList.LinkedList和Vector源码分析 ArrayList ArrayList是一个底层使用数组来存储对象,但不是线程安全的集合类 ArrayList的类结构关系 public ...
- day01-课后作业
#1.写一个登陆的程序,最多登陆失败3次#2.输入账号 密码,如果登录成功,程序结束,提示 欢迎 xx 登录,今天的日期是 xx#3.登录失败,重新登陆#3.要判断输入是否为空,什么也不输入,输入空格 ...
- 33.1 File 获取目录下的所有文件及子目录
重要获取功能 String[] list() 返回当前路径下所有的文件和文件夹名称 //注意:只有指向文件夹的File对象才可以调用该方法(指向文件的file对象使用list会报错npe) File[ ...
- 【FreeMarker】【程序开发】数据模型,对象包装
[FreeMarker][程序开发]数据模型,对象包装 分类: Java.FreeMarker2014-10-25 18:49 413人阅读 评论(0) 收藏 举报 FreeMarker 目录(? ...
- 表字段或表名出现Mysql关键字或保留字导致问题 Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException: You have
MySQL 5.7使用的关键字和保留字 https://dev.mysql.com/doc/refman/5.7/en/keywords.html 当我们建表的时候如果使用了关键字或者保留字,则在执行 ...
- Docker-Bridge Network 02 容器与外部通信
本小节介绍bridge network模式下,容器与外部的通信. 1.前言2.容器访问外部2.1 访问外网2.2 原理2.3 一张图总结2.4 抓包3.外部访问容器3.1 创建nginx容器并从外部访 ...
- 用Python绘制全球疫情变化地图
目前全球疫情仍然比较严重,为了能清晰地看到疫情爆发以来至现在全球疫情的变化趋势,我绘制了一张疫情变化地图,完整代码共 230 行,需要的朋友在公众号回复关键字 疫情地图 即可. 废话不多说,先上图 下 ...
- Redis安装部署(一主二从三哨兵)
需求:根据当前客户的生产环境,模拟安装部署Redis的测试环境,方便后续的功能测试. 1.准备工作 2.安装编译Redis 3.Redis运行环境配置 4.Redis启动和关闭 1.准备工作 Redi ...