想要精通算法和SQL的成长之路 - 戳气球

2023-10-29

想要精通算法和SQL的成长之路 - 戳气球

前言

想要精通算法和SQL的成长之路 - 系列导航

一. 戳气球

原题链接
在这里插入图片描述

首先我们看一下题干:对于超出了数组边界的,就当做它是一个数字为1的气球。遇到这种的,我们可以考虑给数组边界添加哨兵。其值为1。

// 左右各加一个哨兵节点
public int maxCoins(int[] nums) {
    // 1.先处理特殊情况
    if (nums.length == 1) {
        return nums[0];
    }
    int len = nums.length;
    // 左右各加一个哨兵节点
    int[] arr = new int[len + 2];
    arr[0] = 1;
    arr[len + 1] = 1;
    for (int i = 1; i < len + 1; i++) {
        arr[i] = nums[i - 1];
    }
    return ???
}

其次,我们假设有这么一个函数dfs,代表在(left,right)开区间内,可以获得的最大硬币数量。为啥是开区间?因为我们为数组添加了两个哨兵,这俩哨兵不应该在处理范围内。同时,设置俩哨兵的意义也就是:我们认定,每次戳气球的时候,必定存在至少3个气球(有两个可能是哨兵气球)

// nums 在(left,right)开区间内戳气球
public int dfs(int[] nums, int left, int right) {
}

那么,dfs递归,我们首先要写的就是他的递归终止条件:

// 气球数量不足3个
if (right - left < 2) {
    return 0;
}
// 气球数量正好3个
if (right - left == 2) {
	// 我们只能戳破中间的气球(左右两侧是哨兵),获得对应硬币数量
    return nums[left] * nums[left + 1] * nums[right];
}

其次,dfs递归,我们做啥事情?根据前面的递归终止条件可以判断出,此时气球的数量必定 > 3个。我们假设戳气球的步骤:

  1. 先把第k个气球的左侧给戳破完,那么左侧区域能获得的最大硬币数量为:dfs(nums, left,k)
  2. 再把第k个气球的右侧给戳破完,那么右侧区域能获得的最大硬币数量为:dfs(nums, k,right)
  3. 最后戳破第k个气球,那么戳破当前气球获得的硬币数量为:nums[k] * nums[left] * nums[right]

那么写成代码就是:

// nums 在(left,right)开区间内戳气球
public int dfs(int[] nums, int left, int right) {
    int res = 0;
    // 气球数量不足3个
    if (right - left < 2) {
        return 0;
    }
    // 气球数量正好3个
    if (right - left == 2) {
        return nums[left] * nums[left + 1] * nums[right];
    }
    // 在[left+1,right-1]闭区间内遍历,选取每次遍历的当前节点作为 最后戳破 的气球(最后戳破。最后戳破。最后戳破)
    for (int k = left + 1; k < right; k++) {
        // 戳破第k个气球,该气球左侧的最大硬币数
        int leftCount = dfs(nums, left, k);
        int rightCount = dfs(nums, k, right);
        int currentFinalCount = nums[k] * nums[left] * nums[right];
        res = Math.max(res, leftCount + rightCount + currentFinalCount);
    }
    return res;
}

最终的完整代码:

public int maxCoins(int[] nums) {
    // 1.先处理特殊情况
    if (nums.length == 1) {
        return nums[0];
    }
    int len = nums.length;
    // 左右各加一个哨兵节点
    int[] arr = new int[len + 2];
    arr[0] = 1;
    arr[len + 1] = 1;
    for (int i = 1; i < len + 1; i++) {
        arr[i] = nums[i - 1];
    }
    return dfs(arr, 0, len + 1);
}

// nums 在(left,right)开区间内戳气球
public int dfs(int[] nums, int left, int right) {
    int res = 0;
    // 气球数量不足3个
    if (right - left < 2) {
        return 0;
    }
    // 气球数量正好3个
    if (right - left == 2) {
        return nums[left] * nums[left + 1] * nums[right];
    }
    // 在[left+1,right-1]闭区间内遍历,选取每次遍历的当前节点作为 最后戳破 的气球(最后戳破。最后戳破。最后戳破)
    for (int k = left + 1; k < right; k++) {
        // 戳破第k个气球,该气球左侧的最大硬币数
        int leftCount = dfs(nums, left, k);
        int rightCount = dfs(nums, k, right);
        int currentFinalCount = nums[k] * nums[left] * nums[right];
        res = Math.max(res, leftCount + rightCount + currentFinalCount);
    }
    return res;
}

1.1 记忆化搜索

填充书架 一样,我们在dfs递归的时候,有大量的重复计算,我们用一个全局的dfsCache做一下缓存即可。

dfsCache的初始化:

dfsCache = new int[len + 2][len + 2];
for (int i = 0; i < len + 2; i++) {
    Arrays.fill(dfsCache[i], -1);
}

dfsCache的作用体现:如果发现计算过这个值,直接返回,后续的递归直接不用做了。

if (dfsCache[left][right] != -1) {
    return dfsCache[left][right];
}

完整代码:

public class Test312 {
    int[][] dfsCache;

    public int maxCoins(int[] nums) {
        // 1.先处理特殊情况
        if (nums.length == 1) {
            return nums[0];
        }
        int len = nums.length;
        // 左右各加一个哨兵节点
        int[] arr = new int[len + 2];
        arr[0] = 1;
        arr[len + 1] = 1;
        for (int i = 1; i < len + 1; i++) {
            arr[i] = nums[i - 1];
        }
        dfsCache = new int[len + 2][len + 2];
        for (int i = 0; i < len + 2; i++) {
            Arrays.fill(dfsCache[i], -1);
        }
        return dfs(arr, 0, len + 1);
    }

    // nums 在(left,right)开区间内戳气球
    public int dfs(int[] nums, int left, int right) {
        int res = 0;
        // 气球数量不足3个
        if (right - left < 2) {
            return 0;
        }
        // 气球数量正好3个
        if (right - left == 2) {
            return nums[left] * nums[left + 1] * nums[right];
        }
        if (dfsCache[left][right] != -1) {
            return dfsCache[left][right];
        }
        // 在[left+1,right-1]闭区间内遍历,选取每次遍历的当前节点作为 最后戳破 的气球(最后戳破。最后戳破。最后戳破)
        for (int k = left + 1; k < right; k++) {
            // 戳破第k个气球,该气球左侧的最大硬币数
            int leftCount = dfs(nums, left, k);
            int rightCount = dfs(nums, k, right);
            int currentFinalCount = nums[k] * nums[left] * nums[right];
            res = Math.max(res, leftCount + rightCount + currentFinalCount);
        }
        return dfsCache[left][right] = res;
    }

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

想要精通算法和SQL的成长之路 - 戳气球 的相关文章

  • SQLite (Android):使用 ORDER BY 更新查询

    Android SQLite 我想要在 myTable 中的其他行之间插入行在android中使用SQLite 为此 我尝试增加从第 3 行开始的所有行的 id 这样 我就可以在位置 3 处插入新行 myTable 的主键是列 id 表中没
  • 如何引用下一行的数据?

    我正在 PostgreSQL 9 2 中编写一个函数 对于股票价格和日期的表 我想计算每个条目较前一天的百分比变化 对于最早一天的数据 不会有前一天 因此该条目可以简单地为 Nil 我知道WITH声明可能不应该高于IF陈述 到目前为止 这就
  • TSQL - 生成文字浮点值

    我理解比较浮点数时遇到的许多问题 并对它们在这种情况下的使用感到遗憾 但我不是表格作者 只有一个小障碍需要克服 有人决定使用浮点数 就像您期望使用 GUID 一样 我需要检索具有特定浮点值的所有记录 sp help MyTable Colu
  • 标量子查询包含多行

    我正在使用 H2 数据库并想要移动一些数据 为此 我创建了以下查询 UPDATE CUSTOMER SET EMAIL SELECT service EMAIL FROM CUSTOMER SERVICE AS service INNER
  • 分组和切换列和行

    我不知道这是否会被正式称为枢轴 但我想要的结果是这样的 Alex Charley Liza 213 345 1 23 111 5 42 52 2 323 5 23 1 324 5 我的输入数据采用这种形式 Apt Name
  • 根据由另一列分组的不同列的最大值获取值[重复]

    这个问题在这里已经有答案了 我想根据由另一列分组的不同列的最大值来获取列的值 我有这张表 KEY NUM VAL A 1 AB B 1 CD B 2 EF C 2 GH C 3 HI D 1 JK D 3 LM 并想要这样的结果 KEY V
  • 需要 SQL 查询澄清[关闭]

    Closed 这个问题不符合堆栈溢出指南 help closed questions 目前不接受答案 我有一个由以下列组成的表 billid patientid doctorid fees 如何显示治疗多名患者的医生 尝试了以下代码并得到了
  • 如何将SQL数据加载到Hortonworks中?

    我已在我的电脑中安装了 Hortonworks SandBox 还尝试使用 CSV 文件 并以表结构的方式获取它 这是可以的 Hive Hadoop nw 我想将当前的 SQL 数据库迁移到沙箱 MS SQL 2008 r2 中 我将如何做
  • 更好地理解 SQL Server 中的架构

    就像标题一样 我还是一个SQLServer菜鸟 当我创建表 Mytable 时 数据库中显示 dbo Mytable 但有人能让我更好地理解模式吗 另外 在 Server 2008 TSQL 一书中 Itzik 说 在你的数据库中 表属于模
  • Java、Oracle 中索引处缺少 IN 或 OUT 参数:: 1 错误

    您好 我使用 Netbeans 8 0 2 和 Oracle 11g Express Edition 在 JSF 2 2 中编写了一个图书馆管理系统 我有几个名为 书籍 借阅者 等的页面 以及数据库中一些名为相同名称的表 我的问题是这样的
  • SQL - 需要查找重复记录但排除反向事务

    我有一张交易表 偶尔会有 重复条目 如果 当管理员发现这些重复条目时 他们将撤销交易 从而创建负值 但由于监管要求 原始重复条目仍然保留 我想创建一个 SQL 查询 并使用 Crystal Reports 来制作报告 以便管理员轻松查找重复
  • 更改mysql数据库表中的日期格式

    大家早上好 只是一个简单的问题 在我现有的 MySql 数据库中 我几乎没有包含日期 的列 目前这些是年 月 日格式 但现在我需要将其全部更改为年 月 日格式 我试过了select date format curdate d m Y 但它不
  • postgresql 不同的不工作

    我使用以下代码从数据库获取值 但是当我编写这段代码时 测试看看问题出在哪里 我注意到查询没有从数据库中获取不同的值 这是查询 select distinct ca id as id acc name as accName pIsu name
  • meta_query,如何使用关系 OR 和 AND 进行搜索?

    已解决 请参阅下面的答案 我有一个名为的自定义帖子类型BOOKS 它有几个自定义字段 名称为 TITLE AUTHOR GENRE RATING 我该如何修复我的meta query下面的代码以便仅books在自定义字段中包含搜索词 tit
  • 解析错误:语法错误,意外的 T_RETURN [关闭]

    这个问题不太可能对任何未来的访客有帮助 它只与一个较小的地理区域 一个特定的时间点或一个非常狭窄的情况相关 通常不适用于全世界的互联网受众 为了帮助使这个问题更广泛地适用 访问帮助中心 help reopen questions 遇到这个问
  • ALTER TABLE 语句与 FOREIGN KEY 约束冲突

    为什么要添加外键tblDomare表导致此错误 ALTER TABLE 语句与 FOREIGN KEY 约束 FK tblDomare PersN 5F7E2DAC 冲突 冲突发生在数据库 almu0004 表 dbo tblBana 列
  • 支持 >65k 行的 Excel VBA SQL 驱动程序

    在 Excel 2010 中通过 VBA 查询 Excel 数据时 我遇到一个有趣的问题 我正在使用这些驱动程序连接到 xls 或 xls x m 文件 Sub OpenCon ByRef theConn As Connection ByV
  • SQL Server 查询中 UNION ALL 与 OR 条件

    我必须根据表上不存在的条件选择一些行 如果我使用如下的 union all 它会在不到 1 秒的时间内执行 SELECT 1 FROM dummyTable WHERE NOT EXISTS SELECT 1 FROM TABLE t WH
  • 使用用户定义函数 MySql 时出错

    您好 请帮我解决这个问题 提前致谢 我在数据库中定义了这些函数 CREATE FUNCTION levenshtein s1 VARCHAR 255 s2 VARCHAR 255 RETURNS INT DETERMINISTIC BEGI
  • 多边形内的 SQL 地理点在 STIntersect 上不返回 true(但使用 Geometry 返回 true)

    我不想仅仅为了在 STIntersect 中返回 true 而将地理数据转换为几何图形 下面是 SQL 中的代码 DECLARE point GEOGRAPHY GEOGRAPHY Point 1 1 4326 DECLARE polygo

随机推荐

  • 第十章:C语言的调试

    很多小伙伴刚开始听到C语言的调试 这是个啥 表示很怀疑 敲代码不就是直接就是干嘛 结果很多小白们 一运行错误多的数都数不过来 就开始这改改 那删删 莫名奇妙就运行成功了 到最后都不知道到底那错了 有一种小朋友是否有多问号的感觉 可想而知代码
  • AndroidStudio历史记录找回本地修改代码

    今天干了一件特别无语的事情 在现有项目中将新需求开发的代码包想挪一下位置 结果在AndroidStudio中移动失败了 并且原有的包下所有代码都找不到了 搜索了整个项目都没有找到相应的java文件 瞬间心慌啊 这意味着该包下的代码白写了 想
  • npm install报错 -> npm ERR! Unexpected token ‘.‘ 报错解决办法

    原因 我遇到这个问题的场景是用nvm1 1 7的版本安装了16 x以上的node 然后再下载依赖的时候就报错了 总结一下就是nvm版本太低了 他的里面没有集成高版本node导致的 解决 我们把nvm版本换到最新的就可以了 1 卸载掉当前所有
  • 第4章 R语言编程基础——数据整理与预处理

    目录 4 1 经济 金融数据库 4 1 1 金融数据与数据库 4 1 2 国外金融数据库概况 4 1 3 国内金融数据库概况 4 1 4 数据的主要内容 4 2 数据格式 4 3 数据的导入 4 3 1 从控制台上输入数据 4 3 2 上市
  • 异步模式之生产者与消费者

    1 定义 异步 由于存在消息队列 生产者产生的数据不能立刻被消费者处理 中间会有延迟 因此归为异步 异步与同步的区别 同步 线程A要请求某个资源 但是此资源正在被线程B使用中 因为同步机制存在 线程A请求不到 只能等待下去 异步 线程A要请
  • 存储过程违反GTID一致性的问题解决方法

    java sql SQLException Statement violates GTID consistency CREATE TEMPORARY TABLE 解决 2021 2 1项目现场反馈存储过程程序报错无法创建和删除临时表 语句违
  • io.netty.channel.AbstractChannel$AnnotatedConnectException: Connection refused: no further informati

    项目启动报错 io netty channel AbstractChannel AnnotatedConnectException Connection refused no further information 127 0 0 1 63
  • Linux访问ioctl访问失败的问题

    今天遇到一个ioctl访问失败的问题 做个记录 主要是用户态是32位 内核态时64位的 对于字符设备 内核中ioctl的挂接有不同 一 写64位driver驱动时 必须实现compat ioctl实现 用户态是32位时 会调用这个接口 否则
  • SpringBoot使用多线程

    一 概述 1 为什么使用多线程 在我们开发系统过程中 经常会处理一些好费时间的任务 如 向数据库中插入上百万数据 将会导致系统等待 这个时候就会自然想到使用多线程 2 为什么使用Spring来实现多线程 使用Spring比使用JDK原生的并
  • 企业人员信息管理(一)

    一 struts2和hibernate整合 1 整合 StrutsPrepareAndExecuteFilter作用详解 https blog csdn net clk esunny article details 80293978 过滤器
  • 跨境外贸业务,选择动态IP还是静态IP?

    在跨境业务中 代理IP是一个关键工具 它们提供了匿名的盾牌 有助于克服网络服务器针对数据提取设置的限制 无论你是需要经营管理跨境电商店铺 社交平台广告投放 还是独立站SEO优化 代理IP都可以让你的业务程度更加丝滑 达到事半功倍的效果 代理
  • python是不是面向对象的程序设计语言是_Python是一种面向对象程序设计语言

    Python是一种面向对象程序设计语言 答 正确 中国大学MOOC 构建人类命运共同体 要求在政治上 答 相互尊重 平等协商 成人膀胱空虚时膀胱尖不超过 答 耻骨联合上缘 课堂教学中常采用 读一读 议一议 练一练 讲一讲 的教学方式 这符合
  • 百度文心一言可以接入微信小程序啦!

    文心一言 英文名 ERNIE Bot 是百度全新一代知识增强大语言模型 文心大模型家族的新成员 能够与人对话互动 回答问题 协助创作 高效便捷地帮助人们获取信息 知识和灵感 接入小程序效果图 1 百度智能云 千帆大模型平台 注册登录账号 2
  • Qt Creator增强套装16.9.27.12更新

    HI 大家好 这里是jiangcaiyang 我们很高兴地告诉大家 我们将要发布Qt Creator增强套装新的版本了 这一次呢 主要是应大家强烈的要求 更新了我们的聊天神器 萌梦聊天室 现在它不再频繁地崩溃以及暂时性地无法回消息了 这个聊
  • docker安装seata

    下载seata docker镜像 docker pull seataio seata server 1 4 2 创建挂载目录和文件 mkdir p opt docker seata conf touch opt docker seata c
  • 创建老版本react-native项目,以0.59.10为例(0.60.0之前的版本)

    目录 创建react native 0 59 10版本项目前言 开始创建react native 0 59 10版本 创建react native 0 59 10版本项目前言 写这篇文章之前 有些东西要说明一下 当前rn的最新版本为 0 7
  • JavaFx转换为exe

    要点 首先导入依赖 在pom xml导入依赖 具体解释 而maven的两种方式 前者生成两个文件 程序jar包与复制所需的依赖jar包到lib目录 操作比较繁琐 而且在exe4j中进行打包的话会出现Caused by java lang N
  • JS基础知识(二十八):箭头函数

    1 箭头函数的使用 箭头函数有两种格式 一种只包含一个表达式 没有 和 return 一种包含多条语句 这个时候 return 就不能省略 箭头函数类型 代码 没有参数 gt 100 function return 10 一个参数 x gt
  • 使用Initramfs挂载根文件系统,编译过程multiple target patterns(多个目标匹配)问题的解决

    编译内核前 配置内核用Initramfs挂载根文件系统 配置选项如下 Genera setup gt Initial RAM filesystem and RAM disk initramfs initrd support home myr
  • 想要精通算法和SQL的成长之路 - 戳气球

    想要精通算法和SQL的成长之路 戳气球 前言 一 戳气球 1 1 记忆化搜索 前言 想要精通算法和SQL的成长之路 系列导航 一 戳气球 原题链接 首先我们看一下题干 对于超出了数组边界的 就当做它是一个数字为1的气球 遇到这种的 我们可以