N个鸡蛋放入M个篮子且能组合出1~N所有的数

貌似是一道面试题,要求篮子不能为空,且要求出所有可能的放法。

如8个鸡蛋放入4个篮子,可以放 1,1,2,4;也可以放 1,2,2,3。


分析:
  • 不妨以递增的次序来排放鸡蛋,即若第一次放1个,第二次至少放1个。
  • 在这个限制下,设前m个篮子放了n个鸡蛋,那么这m个篮子要能组合出1~n内的所有数。
反证法:假设无法凑出 x,有1<x<n,若x<f(m),则永远无法凑出x,矛盾;若f(m)<x<n,则
一定存在y<f(m) 是无法凑出的,因为存在k使得 x-f(m)-f(m-1)-...-f(k+1)<f(k),该不等式是显然的,
左边的数记为y,于是y<f(k)且y无法用前k个数凑出,因此y永远也无法凑出。上述结论得证。
注:f(i)表示第i个篮子放的鸡蛋数
  • 有了该结论,前m个篮子能组合出1~n,那么第m+1个篮子的取值范围为 [f(m),n],左边界很好理解,右边界设为 v,则使用v的情况下覆盖区域是 [1+v, n+v] ;它和[1, n] 取并集应该覆盖[1, n+v],即1+v<=n+1得到 右边界为 n。
在上述分析下可以进行dfs+剪枝,剪枝会十分高效。

剪枝:

设前m个篮子放了总共n个鸡蛋,第m个篮子放了f(m)个鸡蛋,我们来列举一下第 m+1 个篮子鸡蛋数的剪枝条件:
  • 第m+1个篮子的取值范围是 [ f(m),n ]
  • 假设放了f(m+1)个,后面还有(M-m-1)个篮子,(N-n)个鸡蛋,必须满足 (M-m-1)*f(m+1) <= N-n,即鸡蛋要足够多,使得后面篮子至少能放f(m+1)个鸡蛋;若不满足则说明f(m+1)以及比该值大的放法都不可取;
  • 同样的,还要满足 2^(M-m-1)*f(m+1)>=N,即篮子要足够多;否则,说明f(m+1)取值过小,导致鸡蛋太多没地儿放,因此该f(m+1)及比其小的放法也不可取。
这三个剪枝条件能得到f(m+1)的三个取值区间,取交集就是第m+1个篮子的放法。

mandycool原创,
转载请注明出处,
谢谢!

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值