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 ...
随机推荐
- GlusterFS 4.1 版本选择和部署
GlusterFS 4.1 版本选择和部署 目录 GlusterFS 4.1 版本选择和部署 1 前言相关 1.1 glusterfs优势 1.2 版本选择 1.3 volume知识 2 服务部署 2 ...
- Google GMS介绍
Google GMS介绍GMS全称为GoogleMobile Service.GMS目前提供有Search.Search by Voice.Gmail.Contact Sync.Calendar Sy ...
- MAC 系统java开发环境搭建教程
1.在安装JDK之前,先查看下自己电脑是否已经安装了JDK. 打开终端,输入java -version并回车. 从上图中可以看出我们已安装了,JDK 8.如果这个版本是你需要的版本,可直接看4 ...
- java解惑之常常忘记的事
java解惑之常常忘记的事 2012-10-17 18:38:57| 分类: JAVA | 标签:基础知识 软件开发 |举报|字号 订阅 针对刚接触java的菜鸟来说,java基础知识 ...
- 06-移动web之flex布局
一.基本概念 flex布局又叫伸缩布局 .弹性布局 .伸缩盒布局 .弹性盒布局 Flex 是 Flexible Box 的缩写,意为"弹性布局",用来为盒状模型提供最大的灵活性. ...
- 利用 Github 网络钩子实现自动化部署
GitHub 的网络钩子(webhook)功能,可以很方便的实现自动化部署.本文记录了使用 Node.js 的开发部署过程,当项目的 master 分支被推时,将在服务器进行自动部署 添加网路钩子 在 ...
- 微信群里一道六年级数学题,求阴影面积,那我只能用python代码了
前言 文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. PS:如有需要Python学习资料的小伙伴可以加点击下方链接自行获取http ...
- docker 概览 (1)
Docker Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的镜像中,然后发布到任何流行的 Linux或Windows 机器上,也可以实现虚拟化.容器是完全使 ...
- Chrome插件安利!可以一键导出微信读书笔记|支持Markdown等三种格式
众所周知,微信读书App 是一款非常优秀的阅读类App ,周围也有不少人在用.虽然工作比较忙.但是也没少在上面看书做笔记. 美中不足的是,目前微信读书虽然支持笔记导出,但是提供的是将笔记复制到剪切板, ...
- vue2.x学习笔记(一)
使用vue开发项目已经过了一段时间了,对其中的很多东西还是一知半解,于是想要系统学习一下.主要内容是参照官方中文网站https://cn.vuejs.org/v2/guide/,然后加上一些自己的理解 ...