leetcode-887-鸡蛋掉落(包含最大值最小化,最小值最大化的二分优化+滚动数组的原理)

这里写目录标题

  • 题意
  • 解题
    • KNN复杂度DP解法思想(超时)
    • 上述方法的优化 (最大值最小化二分优化)
      • 完整代码
    • 逆向思维的DP(ksqrt(n)复杂度)
      • 代码
        • 空间优化(滚动数组)
        • 代码

题意

链接:leetcode-887-鸡蛋掉落
给你 k 枚相同的鸡蛋,并可以使用一栋从第 1 层到第 n 层共有 n 层楼的建筑。

已知存在楼层 f ,满足 0 <= f <= n ,任何从 高于 f 的楼层落下的鸡蛋都会碎,从 f 楼层或比它低的楼层落下的鸡蛋都不会破。

每次操作,你可以取一枚没有碎的鸡蛋并把它从任一楼层 x 扔下(满足 1 <= x <= n)。如果鸡蛋碎了,你就不能再次使用它。如果某枚鸡蛋扔下后没有摔碎,则可以在之后的操作中 重复使用 这枚鸡蛋。

请你计算并返回要确定 f 确切的值 的 最小操作次数 是多少?

示例 1:

输入:k = 1, n = 2 输出:2

解释:
鸡蛋从 1 楼掉落。如果它碎了,肯定能得出 f = 0 。
否则,鸡蛋从 2 楼掉落。如果它碎了,肯定能得出 f = 1 。
如果它没碎,那么肯定能得出 f = 2 。
因此,在最坏的情况下我们需要移动 2 次以确定 f 是多少。
示例 2:

输入:k = 2, n = 6 输出:3 示例 3:

输入:k = 3, n = 14 输出:4

解题

KNN复杂度DP解法思想(超时)

  • 表示: dp[i][j]: 表示i个鸡蛋,j层楼的最少移动次数
  • 初始状态: dp[0][:] = dp[:][0]=0, dp[1][h]=h(h从1层到n层:很好理解,1个鸡蛋h层,只能从1层慢慢往上试错)
  • 状态转移: 考虑dp(i, j):i个鸡蛋j层需要最少移动多少次数,那么i个鸡蛋我们考虑从k层仍(k在1~j层),dp(i, j) ----- dp(i, [1~k,k层, k~j])
    要么鸡蛋碎了,要么鸡蛋不碎
    如果碎了,说明鸡蛋破碎的界限f在1~k-1层,我们就不考虑k到j层了,此时鸡蛋碎了一个还有i-1个鸡蛋,所以变成了求i-1个鸡蛋k-1层最少移动多少次
    如果没碎,说明鸡蛋破碎的界限f在k+1~j层,我们就不考虑1到k层了,此时鸡蛋没碎还有i个鸡蛋,所以变成了求i个鸡蛋j-k层最少移动多少次
for j in range(1, n+1):res = float("INF")for h in range(1, j+1):res = min(res, max(dp[i][j-h], dp[i-1][h-1])+1)#加1是因为我们人了一次,进行了一次尝试dp[i][j] = res

上述方法的优化 (最大值最小化二分优化)

res = min(res, max(dp[i][j-h], dp[i-1][h-1])+1)

这是一个最大值最小化问题,可以用二分查找
首先明确一点,dp[i]这一层的值是一个递增的序列dp[i][j] <= dp[i][j+c] (c>0)

  • 针对j-h, h是从1~j, 所以dp[i][j-h] 1<=h<=j,是一个递减的序列
  • 针对h-1, h是从1~j, 所以dp[i][h-1] 1<=h<=j,是一个递增的序列
  • 递减的序列就是dp[i][j-h], 递增的序列就是dp[i-1][h-1],横轴是h的取值

在这里插入图片描述

  • 那么max(dp[i][j-h], dp[i-1][h-1])是什么呢?显然是红框的部分
    在这里插入图片描述
  • 那么min(max(dp[i][j-h], dp[i-1][h-1]))是什么呢,显然是这个最低点,既然h从1~j遍历,从这图清晰看出可以使用二分查找,如果

在这里插入图片描述

  • 如果dp[i][j-mid]>dp[i-1][mid-1], 就是左边的虚线情况,说明最低点在mid右边
  • 如果dp[i][j-mid]<dp[i-1][mid-1], 就是右边的虚线情况,说明最低点在mid左边
  • 很符合二分查找了

在这里插入图片描述

完整代码

class Solution:def superEggDrop(self, k: int, n: int) -> int:"""dp(k, n) n层k个鸡蛋最少需要移动多少次"""dp = [[0]*(n+1) for _ in range(k+1)]dp[1][:] = range(0, n+1)for i in range(2, k+1):for j in range(1, n+1):res = float("INF")# for c in range(1, j+1):#     res = min(res, max(dp[i][j-c], dp[i-1][c-1])+1) kNN复杂度l, r = 1, jwhile(l <= r):m = (l+r) // 2if dp[i][j-m] > dp[i-1][m-1]:l = m + 1res = min(res, dp[i][j-m]) + 1elif dp[i][j-m] < dp[i-1][m-1]:r = m - 1res = min(res, dp[i-1][m-1]) + 1else:res = dp[i-1][m-1] + 1break# res = min(res, max(dp[i][j-m], dp[i-1][m-1])+1)dp[i][j] = resreturn dp[k][n]

在这里插入图片描述

逆向思维的DP(ksqrt(n)复杂度)

原题目求的是k个鸡蛋N层最少需要操作多少次才可以确定f。
转换一下: 求k个鸡蛋最多移动f次,求N的最大值

  • 表示: dp[i][j]: 表示i个鸡蛋,最多移动j次的最大值N

  • 初始状态:

  • 状态转移:

  • dp[k][f]:如果有k个蛋,可以操作f次,所能确定的楼层
    dp[k][f-1]:操作一次后还剩f-1次,蛋碎或者没碎。碎了还有k-1个蛋,没碎还有k个蛋就变成dp[k-1][f-1]
    dp[k][f] = dp[k][f-1] + dp[k-1][f-1] +1
    1:是在某一层掷出,其实我没太懂加1的具体,有点抽象

代码

dp = [[0 for _ in range(n+1)] for _ in range(k+1)]
f = 0
while dp[k][f] < n:f += 1for i in range(1, k+1):dp[i][f] = dp[i][f-1] + dp[i-1][f-1] + 1
return f
空间优化(滚动数组)

dp[i][f] = dp[i][f-1] + dp[i-1][f-1] + 1
第f列只用到了f-1列的数据,所以可以用一维数组

代码
dp = [0 for i in range(k+1)]
f = 0
while dp[k] < n:for i in range(k, 0, -1):dp[i] = dp[i] + dp[i-1] + 1f += 1
return f

至于为什么从后向前遍历:简单解释下

![在这里插入图片描述](https://img-blog.csdnimg.cn/224ab9e1f669459cb27cce9a608e85a4.png

  • 首先明确一点:从dp[k][f],变成dp[k]
  • 每一次for i in range(k, 0, -1): dp[i] = dp[i] + dp[i-1] + 1
    在dp[i] + dp[i-1] + 1还没赋值给dp[i]时,dp[i]保存的时上一轮f,i的值,也就是说
    在dp[i] + dp[i-1] + 1还没赋值给dp[i]时, dp[i]=dp[i][f-1]
    但一旦将dp[i] + dp[i-1] + 1赋值给dp[i]时, dp[i]=dp[i][f], 代表这一轮f的值,上一轮f-1的状态就被覆盖掉了,这也是为什么从后往前遍历的原因

对比一下

dp[i][f] = dp[i][f-1] + dp[i-1][f-1] + 1
dp[i] = dp[i] + dp[i-1] + 1

如果从前向后 ,当i变成i+1时

dp[i+1][f] = dp[i+1][f-1] + dp[i][f-1] + 1dp[i+1] = dp[i+1] + dp[i] + 1
"""
从前往后问题出在这儿的
dp[i]应该对应二维dp[i][f-1]的值现在的dp[i]是已经更新的 dp[i] = dp[i] + dp[i-1] + 1dp[i][f] = dp[i][f-1] + dp[i-1][f-1] + 1 不在单纯是dp[i][f-1]的值了,被覆盖了
"""
"""
我们看看从后向前,i变成i-1
dp[i-2]的值应该对应二维dp[i-2][f-1]
而由于从后向前遍历,dp[i-2]的值并没有变化过,所以这里还是对应的二维数组中dp[i-2][f-1]
"""
dp[i-1][f] = dp[i-1][f-1] + dp[i-2][f-1] + 1dp[i-1] = dp[i-1] + dp[i-2] + 1

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/181113.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

qt6:无法使用setFontColor

问题描述 跟着C开发指南视频学习&#xff0c;但是发现无论是直接使用ui设计&#xff0c;还是纯代码都无法实现变更字体颜色的功能。图中显示&#xff0c;点击颜色控件后&#xff0c;文本框的文字加粗、下划线、斜体等才能设置&#xff0c;但是无法变更颜色。 此文提醒qt sty…

vi vim 末尾编辑按GA 在最后一行下方新增一行编辑按Go

vim 快速跳到文件末尾 在最后一行下方新增一行 移到末尾,并且进入文本录入模式 GA (大写G大写A) 在一般模式(刚进入的模式,esc模式) GA 或 Shift ga 先 G 或 shiftg 到最后一行 然后 A 或 shifta 到本行末尾 并且进入文本录入模式 在最后一行下方新增一行 (光标换行,文字不…

蓝桥杯官网填空题(方格填数)

题目描述 本题为填空题&#xff0c;只需要算出结果后&#xff0c;在代码中使用输出语句将所填结果输出即可。 在 2 行 5 列的格子中填入 1 到 10 的数字。 要求&#xff1a; 相邻的格子中的数&#xff0c;右边的大于左边的&#xff0c;下边的大于上边的。 如下图所示的 …

“一键批量拆分HTML文本,高效整理文件,提升工作效率“

您是否曾经被大量的HTML文本文件困扰&#xff0c;难以找到所需的特定信息&#xff1f;现在&#xff0c;我们向您推荐一款强大的工具&#xff0c;它能够一键拆分HTML文本&#xff0c;让您轻松实现文件整理&#xff0c;提高工作效率&#xff01; 首先&#xff0c;在首助编辑高手…

CANoe新建XML自动化Test Modules

文章目录 1.打开Test Modules2.新建Environment3.新建XML Test Modules4.新建.can文件5.打开XML Test Modules6.新建xml脚本并保存7.编译8.在.can文件写个测试用例9.修改报告格式为HTML10.运行查看报告后面介绍的文章会重复用到这部分,这里单独介绍下,后面不做重复介绍。 1.…

python-在系统托盘显示CPU使用率和内存使用率

一、添加轮子 1.添加托盘区图标库 infi.systray from infi.systray import SysTrayIcon 2.添加图像处理库 Pillow from PIL import Image, ImageDraw, ImageFont 3.添加 psutil 来获取CPU、内存信息 import psutil 二、完整代码 from infi.systray import SysTrayIcon …

使用vue3+vite+elctron构建小项目介绍Electron进程间通信

进程间通信 (IPC) 是在 Electron 中构建功能丰富的桌面应用程序的关键部分之一。 由于主进程和渲染器进程在 Electron 的进程模型具有不同的职责&#xff0c;因此 IPC 是执行许多常见任务的唯一方法&#xff0c;例如从 UI 调用原生 API 或从原生菜单触发 Web 内容的更改。 在 …

如何写复盘报告

复盘报告在it公司中是为了在出现事情后&#xff0c;我们更好的回顾事情的前因后果&#xff0c;定位问题&#xff0c;指定解决措施&#xff0c;并且宣导&#xff0c;让这类事情减少发生的概率。那复盘报告一般怎样写合适呢&#xff1f;下来我们就看看&#xff0c; 一、一般会先…

标签识别中的数据泄露:关键分析

一、介绍 在数据驱动的决策时代&#xff0c;收集、处理和分析数据的过程在从医疗保健到金融&#xff0c;从营销到研究的各个领域都发挥着举足轻重的作用。数据分析的基本步骤之一是正确识别数据集中的标签或类别。然而&#xff0c;这项看似简单的任务可能充满挑战&#xff0c;尤…

zip文件解压缩命令全

zip文件解压缩命令全 入门Zip 用法选项示例语法形式和选项基本语法压缩目录将文件添加到现有压缩文件解压缩文件将 zip 文件解压缩到指定目录列出 zip 文件中的内容将 zip 文件加密将 zip 文件解密将 zip 文件中的文件转成 UTF-8 编码Zip 压缩示例创建新的 zip 压缩文件将文件添…

web前端——HTML+CSS实现奥运五环

web前端——HTMLCSS实现奥运五环 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>Document</titl…

Mac安装DBeaver

目录 一、DBeaver Mac版软件简介 二、下载地址 三、DBeaver连接失败报错 3.1 问题描述 3.2 连接失败问题解决 一、DBeaver Mac版软件简介 DBeaver Mac版是一款专门为开发人员和数据库管理员设计的免费开源通用数据库工具。软件的易用性是它的宗旨&#xff0c;是经过精心设计…

2023 年最值得推荐的 10 款 iPhone 数据恢复软件

iPhone 从来都不是一个简单的打电话电话。它就像一台微型电脑&#xff0c;让我们互相联系、拍照、拍视频、发邮件、看文档、看书。然而&#xff0c;随着它成为日常生活的必需品&#xff0c;我们总是容易因各种原因丢失数据&#xff0c;如删除、恢复出厂设置、iOS 错误、文件同步…

单片机温湿度-光照-DHT11-烟雾气体检测控制系统-proteus仿真-源程序

一、系统方案 本设计采用52单片机作为主控器&#xff0c;液晶1602显示&#xff0c;DHT11温湿度&#xff0c;光照、烟雾气体检测&#xff0c;按键设置报警阀值&#xff0c;蜂鸣器报警。 二、硬件设计 原理图如下&#xff1a; 三、单片机软件设计 1、首先是系统初始化 // // …

C/C++输出字符菱形 2021年3月电子学会青少年软件编程(C/C++)等级考试一级真题答案解析

目录 C/C输出字符菱形 一、题目要求 1、编程实现 2、输入输出 二、算法分析 三、程序编写 四、程序说明 五、运行结果 六、考点分析 C/C输出字符菱形 2021年3月 C/C编程等级考试一级编程题 一、题目要求 1、编程实现 给定一个字符&#xff0c;用它构造一个对角线长…

循环语句--JAVA

循环语句 for循环结构 范例 执行流程 while循环结构 格式 范例 流程 for和while的区别 条件控制语句所控制的自增变量,在for循环结束后,就不可以继续使用了 条件控制语句所控制的自增变量,在while循环结束后,还可以继续使用了 数据类型 基本数据类型 char byte boolean …

register_parameter和register_buffer 详解

在参考yolo系列代码或其他开源代码&#xff0c;经常看到register_buffer和 register_parameter的使用&#xff0c;接下来将详细对他们进行介绍。 1. 前沿 在搭建网络时&#xff0c;我们 自定义的参数&#xff0c;往往不会保存到模型权重文件中&#xff0c;或者成为模型可学习…

操作系统复习(2)进程管理

一、概述 1.1程序的顺序执行 一个具有独立功能的程序独占CPU运行&#xff0c;直至得到最终结果的过程称为程序的顺序执行。 程序的并发执行所表现出的特性说明两个问题 ⑴ 程序和计算机执行程序的活动不再一一对应 ⑵ 并发程序间存在相互制约关系&#xff08;要求共享信息&…

docker-compose 简单部署MySQL Database

docker-compose 简单部署MySQL Database 本博文部署MySQL 并与上篇部署的 Flask进行关联 主博客目录&#xff1a;《从零开始学习搭建量化平台笔记》 文章目录 docker-compose 简单部署MySQL Database部署 MySQLMySQL 开放端口与权限 主项目计划需要搭建一个MySQL 数据库为其他部…

python 深度学习 解决遇到的报错问题8

本篇继python 深度学习 解决遇到的报错问题7-CSDN博客 目录 一、OSError: [WinError 127] 找不到指定的程序。 Error loading "D:\my_ruanjian\conda-myenvs\deeplearning\lib\site-packages\torch\lib\caffe2_detectron_ops.dll" or one of its dependencies. 二、…