A、B两个人在玩猜数字游戏,A随机写了一个数字,在[1,100]区间之内,将这个数字写在了一张纸上,然后B来猜。
如果B猜的数字偏小的话,A会提示:“数字偏小”
一旦B猜的数字偏大的话,A以后就再也不会提示了,只会回答“猜对 或 猜错”
问: 乙至少猜(13)多少次才可以准确猜出这个数字,在这种策略下,乙猜的第一个数字是(13)
解析:
首先阅读题目,一个很重要的信息点就是: 一旦B某次猜的偏大,A就不再提示,此次之后B猜的偏小A也不会再提示,只回答猜对与否 。如果没有这个条件,或者说改变这个条件,改为: 如果B猜的偏大,A会提示B这次猜的偏大 那么相信大家都会给出答案,那就是用二分法,只需要7次就可以保证猜对了。
但是现在的条件变了,如果B猜的偏大,那么不提示,所以我们得出结论就是:如果猜的偏大,只能一个一个往下猜。
答案是13次,猜测序列为14, 27, 39, 50, 60,69, 77, 84, 90, 95, 99,哪一次偏大了,就在该数与上一个数的区间一个个猜,最多13次一定能猜到。下面讲讲解题思路:
刚一看到这道题,熟悉二分查找的同学肯定马上想到要用二分查找来猜,第一个猜50,第二个猜25或者75……可是这样有一个问题,传统的二分查找是需要每次都知道是偏大还是偏小的,但这里一旦偏大,就再也得不到这个信息了。这就导致了在这里如果继续使用这种类似二分查找的方法,最坏情况下的猜测次数分布不均匀。
直觉告诉我们,要使得总猜测次数最少,那就让最坏情况下的猜测次数分布均匀即可。假设最多猜测k次,那么第一个猜的数字应该是k+1,因为若偏大了,则逐一把k, k-1, ……2的共k-1个数猜一遍,最坏的情况是都没猜中,则1必定是正确结果;若偏小了,则继续按照下面的方式猜:
若偏小了,则第二个猜的数字x应该是什么呢?这就要使得若第二次猜偏大了的话,必定能把总共的猜测次数也控制在k次,因此第二个猜的数x跟第一个猜的数k+1之间应当间隔k-1个数,因为这样的话,即使第二个数偏大了,则逐一把x-1,x-2,……k+3的共k-2个数猜一遍,必定能得到答案。因此第二个猜的数字x为2k+1。
依此类推,要把100覆盖,则可列出不等式:(k+1) + k + (K-1) + …… + 2 >= 100,解得k >= 13
与之类似的谷歌扔鸡蛋问题:
假设你有2个鸡蛋,你现在想知道这些鸡蛋的硬度。你家住在100层高的大楼里,现在要在这座大楼上测试鸡蛋的硬度。每个鸡蛋的硬度相同,鸡蛋的硬度定义为:如果鸡蛋从第m层上掉下来没有破裂,而从第m+1层上掉下来就破裂了,那么这个鸡蛋的硬度就是m。某个鸡蛋如果在实验中破裂了就永远的损失了。要求提供一种方案,在最坏情况下做最少次数的实验即可把鸡蛋的硬度检验出来
方法同理 设最坏情况下,最少k次 , 每次从 k楼扔下鸡蛋(这里和猜数字不一样,不能是k+1,因为1楼是否破裂不清楚), 则有 k+k-1+k-2+....+1 >= 100
解得 k >=14 所以,需要14次
再扩展一下,如果是 n个鸡蛋m层大楼最坏情况最少需要几次,动态规划方程为。
W(n, k) = 1 + min{max(W(n -1, x -1), W(n, k - x))} 其中W(1,k) = k(相当于1颗鸡蛋测试k层楼问题),W(0,k) = 0,W(n, 0) = 0