算法46:动态规划专练(力扣198: 打家劫舍 力扣740:删除并获取点数)

news2025/1/16 20:13:25

打家劫舍问题

你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警

给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。

示例 1:

输入:[1,2,3,1]
输出:4
解释:偷窃 1 号房屋 (金额 = 1) ,然后偷窃 3 号房屋 (金额 = 3)。
     偷窃到的最高金额 = 1 + 3 = 4 。

示例 2:

输入:[2,7,9,3,1]
输出:12
解释:偷窃 1 号房屋 (金额 = 2), 偷窃 3 号房屋 (金额 = 9),接着偷窃 5 号房屋 (金额 = 1)。
     偷窃到的最高金额 = 2 + 9 + 1 = 12 。

分析:

这一题看起来是隔一家偷一家的问题。其实不是那么回事。它和爬楼梯花费最小代价是一个性质的题目.

a.  只有1户人家,没到选; 2户人家偷钱多的。

b. 如果是 3户人家,那就要看偷的钱数的累加和了。假设数组为: [2,3,4]. 偷的最大金额就是 2+4 =6;中间隔一家

c. 假设是4户人家 : [3,2,4,10]. 

数值32410
下标0123
偷窃最大金额33713

     偷窃的最大金额是13.

package code04.动态规划专项训练01;

/**
 * 力扣198题 : 打家劫舍问题
 * https://leetcode.cn/problems/house-robber/?envType=study-plan-v2&envId=dynamic-programming
 */
public class Rob_05 {
    
    public int rob(int[] nums)
    {
        if (nums == null || nums.length == 0) {
            return 0;
        }

        //只有1间房
        if (nums.length == 1) {
            return nums[0];
        }

        //2间房,偷金额大的
        if (nums.length == 2) {
            return Math.max(nums[0], nums[1]);
        }

        int n = nums.length;
        //动态规划,普遍现象就是dp表长度要多1个。
        int[] dp = new int[n];
        dp[0] = nums[0];
        dp[1] = Math.max(nums[0], nums[1]);

        for (int i = 2; i < nums.length; i++) {
            dp[i] = Math.max(dp[i-1], dp[i-2] + nums[i]);
        }

        return dp[n-1];
    }

    public static void main(String[] args) {
       // int[] arr ={2,1,1,2};
      //  int[] arr ={1,7,9,4};
        int[] arr ={1,2,3,1};
        Rob_05 ss = new Rob_05();
        System.out.println(ss.rob(arr));
    }
}

   

力扣740:删除并获取点数

给你一个整数数组 nums ,你可以对它进行一些操作。

每次操作中,选择任意一个 nums[i] ,删除它并获得 nums[i] 的点数。之后,你必须删除 所有 等于 nums[i] - 1 和 nums[i] + 1 的元素。

开始你拥有 0 个点数。返回你能通过这些操作获得的最大点数。

示例 1:

输入:nums = [3,4,2]
输出:6
解释:
删除 4 获得 4 个点数,因此 3 也被删除。
之后,删除 2 获得 2 个点数。总共获得 6 个点数。

示例 2:

输入:nums = [2,2,3,3,3,4]
输出:9
解释:
删除 3 获得 3 个点数,接着要删除两个 2 和 4 。
之后,再次删除 3 获得 3 个点数,再次删除 3 获得 3 个点数。
总共获得 9 个点数。

这一题很好玩,可以借鉴词频统计的思想。之前我写过关于词频统计的博客,有兴趣的可以翻一翻。

a. 这一题可能会出现重复值的问题。而且如果我们选取了值为3的元素,那就得把2和4的值都给删除掉了。

b. 词频统计的思想,一旦字母固定,那么在数组中的位置也就固定了。同样的道理,一旦值固定,我们创新新的数组,用这个值对应新数组的下标,那么这个值对应的下标也就固定了。

以[2,2,3,3,3,4] 举例子

转变后数组变成了

无重复数组:                          [0, 1, 2, 3, 4] 

根据无重复数组,算的累加和 [0, 0, 4, 9, 4]

 为什么数组长度为4 ?

根据元素数组的最大值确定数组长度。

package code04.动态规划专项训练01;


/**
 * 力扣 740 : 删除并获取点数
 *  https://leetcode.cn/problems/delete-and-earn/description/?envType=study-plan-v2&envId=dynamic-programming
 */
public class DeleteAndEarn_06
{
    public int deleteAndEarn(int[] nums) {

        //边界值
        if (nums == null || nums.length == 0) {
            return 0;
        }
        //只有1个数
        if (nums.length == 1) {
            return nums[0];
        }

        //找出最大值
        int max = 0;
        for(int i = 0; i < nums.length; i++) {
            max = Math.max(max, nums[i]);
        }

        //这一点不好,根据数组的最大值,确定辅助数组的长度
        int size = max + 1;
        //统计出每个值出现的点数
        int[] ss = new int[size];
        for (int i = 0; i < nums.length; i++) {
            //词频统计的思路,每个字母出现对应的ASSC码不变
            //本题中,重复值出现的话,ss数组的下标志值也不会改变,但是点数会累加
            ss[nums[i]] += nums[i];
        }

        //dp表放置的是逐个数出现的最大值。它是随着数组个数的变化而逐渐变化的
        // 数组为1,那最大值就为1.
        //数组为 {1,2}, 那最大值就为2
        //数组为{1,2,4}. 那最大值就是 2+ 4 = 6.
        int[] dp = new int[ss.length];
        dp[0] = ss[0];
        dp[1] = ss[0] >= ss[1] ? ss[0] : ss[1];
        for (int index = 2; index < ss.length; index++) {
            //ss[i]就是有序的,它的排序是按照原始数组的值进行排序的。
            //原始数组一旦选定num[i], 还得考虑num[i]-1 和 num[i]+1 的问题
            // 经过转换:
            // num[i]-1 就变成了 index -1;
            //当前只遍历到 index,不存在 num[i]+1 即 index + 1的情况
            dp[index] = Math.max(dp[index-1], dp[index-2] + ss[index]);
        }

        return dp[max];
    }

    public static void main(String[] args) {
        //int[] arr = {3,4,2};
       //int[] arr = {3,1};
        int[] arr = {1,1,1,2,4,5,5,5,6};
        DeleteAndEarn_06 ss = new DeleteAndEarn_06();
        System.out.println(ss.deleteAndEarn(arr));
    }
}

本道题是数据量可能有限,我看测试是通过了,并设 85%的胜率也还算不错了。

但是,这一题确定新数组长度的思想是按照数组最大值确定的。假设数组为 [2,2,3,3,100万], 难到你的新数组长度要定义成100万?

离散化技巧 : 在说线段树的时候,我多次使用过离散化的技巧。并且专门分析过:算法41:掉落的方块(力扣699题)----线段树-CSDN博客

这一题,我觉得使用离散化技巧去确定新数组的长度更合适,具体我就不去尝试了,感谢的可以尝试一下

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

flutter框架优缺点,最新Android大厂高频面试题

部分面试常问的面试专题 一、Java篇 1.多线程并发&#xff1b; sleep 和 wait 区别join 的用法线程同步&#xff1a;synchronized 关键字等线程通信线程池手写死锁 2.Java 中的引用方式&#xff0c;及各自的使用场景 3.HashMap 的源码 4.GC(垃圾回收)是什么&#xff1f;如何…

运维随录实战(8)之搭建数据库

docker搭建mysql5.7 如果出现no matching manifest for windows/amd64 10.0.18363 in the manifest list entries错误,则将docker desktop的settings->Docker Engine中的experimental值设为true。 touch : 无法将“touch”项识别为 cmdlet、函数、脚本文件或可运行程序的…

is not valid JSON at JSON.parse

在后台读取一个文件里的JSON数据&#xff0c;转换成字符串返回给前端&#xff0c;前端使用JSON.parse转换JSON报错。在将JSON校验和压缩后发现前端还是转换失败。在返回结果的时候可以看见一个小红点 最后排查&#xff0c;不带BOM的识别是Java遗留的一个bug。 解决方案&#…

【排序】详解插入排序

一、思想 插入排序是通过构建有序序列&#xff0c;对于未排序数据&#xff0c;在已排序序列中从后向前扫描&#xff0c;找到相应位置并插入。具体步骤如下&#xff0c;将数组下标为0的元素视为已经排序的部分&#xff0c;从1开始遍历数组&#xff0c;在遍历的过程中当前元素从…

CSS的文本样式属性值,web前端开发规范

正文 介绍下半连接队列 服务器第一次接收到客户端的SYN后&#xff0c;会处于SYN-REVD阶段&#xff0c;此时双方还没有建立完全的连接&#xff0c; 服务器会把此种状态下请求连接放在一个队列里&#xff0c;我们把这种队列称为半连接队列 已经完成三次握手并建立连接&#xff…

springboot+xjar加密打包部署教程

需求背景 为了跟上时代的步伐&#xff0c;为了更好的生存。开个玩笑&#xff0c;就是心血来潮&#xff0c;使用xjar加密部署jar包&#xff0c;于是就测试一下。 xjar教程 1-maven配置文件修改 首先找到自己ideal配置的maven文件夹&#xff0c;然后找到apache-maven-3.9.3\co…

「爬虫职海录」三镇爬虫

HI&#xff0c;朋友们好 「爬虫职海录」第三期更新啦&#xff01; 本栏目的内容方向会以爬虫相关的“岗位分析”和“职场访谈”为主&#xff0c;方便大家了解一下当下的市场行情。 本栏目持续更新&#xff0c;暂定收集国内主要城市的爬虫岗位相关招聘信息&#xff0c;有求职…

uniapp微信小程序获取当前位置

uni-app微信小程序uni.getLocation获取位置&#xff1b;authorize scope.userLocation需要在app.json中声明permission&#xff1b;小程序用户拒绝授权后重新授权-CSDN博客

HTML实体字符列表,知识点详解

css盒模型 1&#xff0c;css盒模型基本概念&#xff1f; 2&#xff0c;标准模型和IE模型的区别&#xff1a;计算高度和宽度的不同&#xff0c;怎么不同&#xff0c;高度宽度是怎么计算的&#xff1f; 3&#xff0c;css如何设置这两种模型&#xff1f; 4&#xff0c;js如何设置…

uniapp实现---类似购物车全选

目录 一、实现思路 二、实现步骤 ①view部分展示 ②JavaScript 内容 ③css中样式展示 三、效果展示 四、小结 注意事项 一、实现思路 点击商家复选框&#xff0c;可选中当前商家下的所有商品。点击全选&#xff0c;选中全部商家的商品 添加单个多选框&#xff0c;在将多选…

javaWebssh题库管理系统myeclipse开发mysql数据库MVC模式java编程计算机网页设计

一、源码特点 java ssh题库管理系统是一套完善的web设计系统&#xff08;系统采用ssh框架进行设计开发&#xff09;&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。开发环境为TOMCAT7.0,Mye…

第七篇:人工智能与机器学习技术VS量测(Measurement)- 我为什么要翻译介绍美国人工智能科技巨头IAB公司 - 它是如何赋能数字化营销生态的?

IAB平台&#xff0c;使命和功能 IAB成立于1996年&#xff0c;总部位于纽约市。 作为美国的人工智能科技巨头社会媒体和营销专业平台公司&#xff0c;互动广告局&#xff08;IAB- the Interactive Advertising Bureau&#xff09;自1996年成立以来&#xff0c;先后为700多家媒…

CSS字体样式值,前端开发基础学习

元素特点&#xff1a; 块状元素&#xff1a; 在页面中以矩形区域显示。自上而下排列&#xff0c;独占一行可以直接添加宽高一般情况下&#xff0c;作为其他元素或内容的容器 行内元素&#xff1a; 在页面中最小单位也是矩形。在一行内逐个排列。不可以直接添加宽高&#xf…

FPGA高端项目:FPGA基于GS2971的SDI视频接收+HLS图像缩放+多路视频拼接,提供4套工程源码和技术支持

目录 1、前言免责声明 2、相关方案推荐本博已有的 SDI 编解码方案本方案的SDI接收转HDMI输出应用本方案的SDI接收图像缩放应用本方案的SDI接收纯verilog图像缩放纯verilog多路视频拼接应用本方案的SDI接收HLS动态字符叠加输出应用本方案的SDI接收HLS多路视频融合叠加应用本方案…

GIT使用学习笔记

最近发现了一个学习git的网站&#xff0c;可以学习git常用的命令&#xff0c;并且可以实操练习以及动画演示&#xff0c;通关以后对git有了非常深入的理解&#xff0c;希望大家也去这个网站里面实际操作一下&#xff0c;我这边仅作笔记&#xff0c;方便自己后续查阅 https://le…

技术选型思考:分库分表和分布式DB(TiDB/OceanBase) 的权衡与抉择

在当今数据爆炸的时代&#xff0c;数据库作为存储和管理数据的核心组件&#xff0c;其性能和扩展性成为了企业关注的重点。随着业务的发展和数据量的不断增长&#xff0c;传统的单库单表架构逐渐暴露出性能瓶颈和扩展性限制。为了应对这些挑战&#xff0c;企业常常需要在分库分…

学习c语言:动态内存管理

一、为什么要有动态内存分配 我们已经掌握的内存开辟⽅式有&#xff1a; int val 20; //在栈空间上开辟四个字节 char arr[10] {0}; //在栈空间上开辟10个字节的连续空间 但是上述的开辟空间的⽅式有两个特点&#xff1a; • 空间开辟⼤⼩是固定的。 • 数组在申明的时候&…

StarUML6.0.1使用

1. 简介 作为一个软件开发人员&#xff0c;平时免不了做一定的软件设计&#xff0c;标准做法就是采用UML来设计&#xff1a; 讨论功能流程时采用时序图、活动图来表达&#xff1b;做业务功能架构时采用组件图来表达&#xff1b;做系统部署架构时采用部署图来表达&#xff1b;做…

python一张大图找小图的个数

python一张大图找小图的个数 一、背景 有时候我们在浏览网站时&#xff0c;发现都是前端搞出来的一张张图&#xff0c;我们只能用盯住屏幕的小眼睛看着&#xff0c;很累的统计&#xff0c;这个是我在项目中发现没办法统计&#xff0c;网上的教程很多&#xff0c;都不成功&…

微信小程序开发系列(十一)·小程序页面的跳转设置以及参数传递

目录 1. 跳转到商品列表 1.1 url: 当前小程序内的跳转链接 1.2 navigate&#xff1a;保留当前页面&#xff0c;跳转到应用内的某个页面。但是不能跳到 tabbar 页面 1.3 redirect&#xff1a; 关闭当前页面&#xff0c;跳转到应用内的某个页面。但不能跳转到 tabbar 页面…