题目描述

这是 LeetCode 上的 808. 分汤 ,难度为 中等

Tag : 「数学」、「动态规划」、「线性 DP」

有 A 和 B 两种类型 的汤。一开始每种类型的汤有 n 毫升。有四种分配操作:

  1. 提供 100ml 的 汤A 和 0ml 的 汤B 。
  2. 提供 75ml 的 汤A 和 25ml 的 汤B 。
  3. 提供 50ml 的 汤A 和 50ml 的 汤B 。
  4. 提供 25ml 的 汤A 和 75ml 的 汤B 。

当我们把汤分配给某人之后,汤就没有了。每个回合,我们将从四种概率同为 0.25 的操作中进行分配选择。如果汤的剩余量不足以完成某次操作,我们将尽可能分配。当两种类型的汤都分配完时,停止操作。

注意 不存在先分配 100 ml 汤B 的操作。

需要返回的值: 汤A 先分配完的概率 +  汤A和汤B 同时分配完的概率 / 2。返回值在正确答案 10^{-5}10−5 的范围内将被认为是正确的。

示例 1:

输入: n = 50输出: 0.62500解释:如果我们选择前两个操作,A 首先将变为空。
对于第三个操作,A 和 B 会同时变为空。
对于第四个操作,B 首先将变为空。
所以 A 变为空的总概率加上 A 和 B 同时变为空的概率的一半是 0.25 *(1 + 1 + 0.5 + 0)= 0.625。
复制代码

示例 2:

输入: n = 100输出: 0.71875
复制代码

提示:

  • 0 <= n <= 10^90<=n<=109

数学 + 动态规划

四种分配方式都是 2525 的倍数,因此我们可以将 nn 进行除以 2525 上取整的缩放操作,并将四类操作等价成:

  1. 提供 4ml 的 汤A 和 0ml 的 汤B 。
  2. 提供 3ml 的 汤A 和 1ml 的 汤B 。
  3. 提供 2ml 的 汤A 和 2ml 的 汤B 。
  4. 提供 1ml 的 汤A 和 3ml 的 汤B 。

定义 f[i][j]f[i][j] 为 汤A 剩余 ii 毫升,汤B 剩余 jj 毫升时的最终概率(概率 = 汤A先分配完的概率 + 汤A和汤B同时分配完的概率 \times 0.5概率=汤A先分配完的概率+汤A和汤B同时分配完的概率×0.5)。

最终答案为 f[n][n]f[n][n] 为最终答案,考虑任意项存在为 00 情况时的边界情况:

  • 若 i = 0i=0 且 j = 0j=0,结果为 0 + \frac{1}{2} = \frac{1}{2}0+21​=21​,即有 f[0][0] = 0.5f[0][0]=0.5
  • 若 i = 0i=0 且 j > 0j>0,结果为 1 + 0 = 11+0=1,即有 f[0][X] = 1f[0][X]=1,其中 X > 1X>1
  • 若 i > 0i>0 且 j = 0j=0,结果为 0 + 0 = 00+0=0,即有 f[X][0] = 0f[X][0]=0,其中 X > 1X>1

其余一般情况为 ii 和 jj 均不为 00,由于四类操作均为等概率,结合题意和状态定义可知:

f[i][j] = \frac{1}{4} \times (f[i - 4][j] + f[i - 3][j - 1] + f[i - 2][j - 2] + f[i - 1][j - 3])f[i][j]=41​×(f[i−4][j]+f[i−3][j−1]+f[i−2][j−2]+f[i−1][j−3])

由于 n = 1e9n=1e9,即使进行了除 2525 的缩放操作,过多的状态数仍会导致 TLE

此时需要利用「返回值在正确答案 10^{-5}10−5 的范围内将被认为是正确的」来做优化(一下子不太好想到):由于四类操作均是等概率,单个回合期望消耗汤 A 的量为 2.52.5,消耗汤 B 的量为 1.51.5。

因此当 nn 足够大,操作回合足够多,汤 A 将有较大的概率结束分配,即当 nn 足够大,概率值会趋向于 11。

我们考虑多大的 nn 能够配合精度误差 10^{-5}10−5 来减少计算量:一个可行的操作是利用上述的 DP 思路 + 二分的方式找到符合精度要求的验算值(不超过 200200)。

Java 代码:

class Solution {public double soupServings(int n) {n = Math.min(200, (int) Math.ceil(n / 25.0));double[][] f = new double[n + 10][n + 10];f[0][0] = 0.5;for (int j = 1; j <= n; j++) f[0][j] = 1;for (int i = 1; i <= n; i++) {for (int j = 1; j <= n; j++) {double a = f[Math.max(i - 4, 0)][j], b = f[Math.max(i - 3, 0)][Math.max(j - 1, 0)];double c = f[Math.max(i - 2, 0)][Math.max(j - 2, 0)], d = f[Math.max(i - 1, 0)][Math.max(j - 3, 0)];f[i][j] = 0.25 * (a + b + c + d);}}return f[n][n];}
}
复制代码

Python 代码:

class Solution:def soupServings(self, n: int) -> float:n = min(200, math.ceil(n / 25))f = [[0] * (n + 10) for _ in range(n + 10)]f[0][0] = 0.5for j in range(1, n + 10):f[0][j] = 1for i in range(1, n + 1):for j in range(1, n + 1):a, b = f[max(i - 4, 0)][j], f[max(i - 3, 0)][max(j - 1, 0)]c, d = f[max(i - 2, 0)][max(j - 2, 0)], f[max(i - 1, 0)][max(j - 3, 0)]f[i][j] = 0.25 * (a + b + c + d)return f[n][n]
复制代码
  • 时间复杂度:O(m^2)O(m2),其中 m = 200m=200 为验算值
  • 空间复杂度:O(m^2)O(m2)

最后

这是我们「刷穿 LeetCode」系列文章的第 No.808 篇,系列开始于 2021/01/01,截止于起始日 LeetCode 上共有 1916 道题目,部分是有锁题,我们将先把所有不带锁的题目刷完。

在这个系列文章里面,除了讲解解题思路以外,还会尽可能给出最为简洁的代码。如果涉及通解还会相应的代码模板。

为了方便各位同学能够电脑上进行调试和提交代码,我建立了相关的仓库:github.com/SharingSour… 。

在仓库地址里,你可以看到系列文章的题解链接、系列文章的相应代码、LeetCode 原题链接和其他优选题解。

808. 分汤 : 挺有意思的 DP 题相关推荐

  1. Java实现 LeetCode 808 分汤 (暴力模拟)

    808. 分汤 有 A 和 B 两种类型的汤.一开始每种类型的汤有 N 毫升.有四种分配操作: 提供 100ml 的汤A 和 0ml 的汤B. 提供 75ml 的汤A 和 25ml 的汤B. 提供 5 ...

  2. 【LeetCode】808.分汤

    题目描述 有 A 和 B 两种类型 的汤.一开始每种类型的汤有 n 毫升.有四种分配操作: 提供 100ml 的 汤A 和 0ml 的 汤B . 提供 75ml 的 汤A 和 25ml 的 汤B . ...

  3. LeetCode 808. 分汤(动态规划)

    文章目录 1. 题目 2. 解题 1. 题目 有 A 和 B 两种类型的汤.一开始每种类型的汤有 N 毫升.有四种分配操作: 提供 100ml 的汤A 和 0ml 的汤B. 提供 75ml 的汤A 和 ...

  4. 三个人分汤终极解答,四人分汤,多人分汤终极答案。

    网上其他的基本都是错误的答案呀..思维不慎密..终极正确答案在这里.. 今天看到道题目,感觉很有意思: 一间囚房里关押着两个犯人.每天监狱都会为这间囚房提供一罐汤,让这两个犯人自己来分.起初,这两个人 ...

  5. 逻辑心理测试题:三囚分汤

    一间囚房里关押着两个犯人.每天监狱都会为这间囚房提供一罐汤,让这两个犯人自己来分.起初,这两个 人经常会发生争执,因为他们总是有人认为对方的汤比自己的多.后来他们找到了一个两全其美的办法:一个人分汤, ...

  6. poj2479与poj2593 , 同一道DP题

    Description Give you N integers a1, a2 ... aN (|ai| <=1000, 1 <= i <= N).  You should outpu ...

  7. 国考省考行测:继续讲结构分析法的分总、总分、分分、分总分、总分总的真题例题讲解,错项的特点规律

    国考省考行测:继续讲结构分析法的分总.总分.分分.分总分.总分总的真题例题讲解,错项的特点规律 2022找工作是学历.能力和运气的超强结合体! 公务员特招重点就是专业技能,附带行测和申论,而常规国考省 ...

  8. 程序猿有话说:计算机,学着挺有意思的,就是头冷

    程序猿,一个互联网高速发(cui)展(can)下的新生物种,不仅拥有较高的智商,还有无限的忍耐力,日常敲代码,加班已成为了常态,一个高薪与秃头并存的群体- 据说程序员的日常是这样的: 当然大家一提起程 ...

  9. 计算机系专用表情包,计算机学起来挺有意思的表情包 - 计算机学起来挺有意思的微信表情包 - 计算机学起来挺有意思的QQ表情包 - 发表情 fabiaoqing.com...

    这世界挺有意思的没意思的是我!(熊猫头)_挺有_没意思_熊猫_意思_世界表情 这世界挺有意思的没意思的是我(熊猫头)_挺有_没意思_熊猫_意思_世界表情 电子城轨通信计算机:快看!那个学会计的上吊了_ ...

最新文章

  1. 车联网技术对中老年人吸引力更大
  2. 60条令你大吃一惊的小常识,很有用
  3. 用python读写excel(xlrd、xlwt)
  4. Leetcode算法题(C语言)12--旋转图像
  5. Microsoft向高性能计算市场推出了新的Azure产品
  6. 宝刀不老: Cookie (转)
  7. 常见数通设备镜像制作模板
  8. 详解CAN 2.0协议
  9. 【HUSTOJ】1053: 字符图形9-数字正三角
  10. linux x86软件改arm,x86程序移植到arm上
  11. 演讲的思路锻炼,逆向思维需要刻意练习吗?
  12. 人工智能原理笔记------知识表示方法
  13. 如何设置vs窗口的属性管理器和解决方案管理器的位置
  14. DWC的1000M的MAC自环和PHY自环测试寄存器修改方式
  15. 技术笔记:Indy IdSMTP支持腾讯QQ邮箱邮件发送
  16. 万字长文带你探究 Docker 容器化技术背后的黑科技
  17. php 怎么验证邮箱验证码,PHP实现邮箱验证码验证功能
  18. 微信真的在后台频繁读取用户相册吗?
  19. bzoj 2733 永无岛
  20. 2022-2028全球与中国平板显示器制造设备市场现状及未来发展趋势

热门文章

  1. Es6基础总结——前端必看
  2. 母亲节快到!我打算买个蛋糕,谢谢妈妈,祝妈妈健康、长寿、快乐!!!!!!!!!!!
  3. 怎么在本地运行umi框架的生产模式
  4. 基本注解详解@RequestMapping,@GetMapping
  5. 细数每日优鲜的五大增长点,是什么让它APP月活突破千万?
  6. 正月初二,大鹏杨梅坑之旅
  7. 记一次hackbar乱码密文中文解码
  8. 20行代码,带你了解未来颠覆性的工作模式
  9. 搭建DM数据库LINUX单机环境
  10. sklearn学习-朴素贝叶斯