【面试高频题】值得仔细推敲的贪心及其证明

news2025/1/22 17:48:58

题目描述

这是 LeetCode 上的 「1846. 减小和重新排列数组后的最大元素」 ,难度为 「中等」

Tag : 「贪心」

给你一个正整数数组 arr

请你对 arr 执行一些操作(也可以不进行任何操作),使得数组满足以下条件:

  1. arr 中 第一个 元素必须为 

  2. 任意相邻两个元素的差的绝对值小于等于 

    对于任意的  ,都满足 abs(arr[i]-arr[i-1])<=1abs(x) 为 x 的绝对值。

你可以执行以下 种操作任意次:

  • 减小 arr 中任意元素的值,使其变为一个更小的正整数

  • 重新排列 arr 中的元素,你可以以任意顺序重新排列

请你返回执行以上操作后,在满足前文所述的条件下,arr 中可能的最大值。

示例 1:

输入:arr = [2,2,1,2,1]

输出:2

解释:
我们可以重新排列 arr 得到 [1,2,2,2,1] ,该数组满足所有条件。
arr 中最大元素为 2 。

示例 2:

输入:arr = [100,1,1000]

输出:3

解释:
一个可行的方案如下:
1. 重新排列 arr 得到 [1,100,1000] 。
2. 将第二个元素减小为 2 。
3. 将第三个元素减小为 3 。
现在 arr = [1,2,3] ,满足所有条件。
arr 中最大元素为 3 。

示例 3:

输入:arr = [1,2,3,4,5]

输出:5

解释:数组已经满足所有条件,最大元素为 5 。

提示:

基本分析 & 证明

根据题意,数组的第一位必须是 ,且每个数只能 「减小」「不变」,数值位置可以任意调整。

求解经过调整后,符合要求的数组中的最大值是多少。

首先符合条件的数组相邻位差值绝对值不超过 ,这限定了数组的必然是如下三种分布之一:

  • (非严格)单调递减
  • 存在波段
  • (非严格)单调递增

证明一:取得最优解对应的数组「必然是」或者「可调整为」(非严格)单调递增的形式。

我们使用反证法来证明另外两种分布不能取得最优解:

  • (非严格)单调递减:题目限定了数的范围为正整数,且第一位为 ,这种情况不用讨论了,跳过;
  • 存在波段:我们始终可以将波峰的右侧出现的值,纳入到波峰的左侧,从而消掉这个波峰,最终将整个分布调整为「(非严格)单调递增」的形式,结果不会变差:
alt

多个波段的情况也是同理,可以自己在纸上画画。

都是利用 波峰右侧的点可以调整成波峰左侧的点,从而使分布变为(非严格)单调递增

至此,我们证明了最优解对应的数组必然符合(非严格)单调递增。

这启发我们可以先对原数组排个序,在此基础上进行分析。

对原数组排序得到的有序数组,不一定是符合「相邻位差值绝对值不超过 」的,同时由于每个数值可以选择 「减小」「不变」

证明二:当必须要对当前位进行调整的时,优先选择调整为「与前一值差值为 的较大数」不会比调整为「与前一差值为 的较小数」更差。

这可以使用归纳推理,假设采取「优先调整为与前一值差值为 的较大数」得到的序列为 a,采用「优先调整与前一差值为 的较小数」得到的序列为 b

根据「 」、「ab 长度一致」、「ab 均为(非严格)单调递增」以及「ab 均满足相邻位差值不超过 」,可推导出 ,和任意位置 ,从而推导出 a 序列的最后一位必然大于等于 b 的最后一位。

b 不会比 a 更优。

证明三:调整大小的操作不会改变数组元素之间的相对位置关系。

在证明二的分析中,我们会对某些元素进行“减小”操作,使得整个数组最终满足「相邻位差值绝对值不超过 」。

但该证明成立的还有一个很重要的前提条件,就是调整操作不会出发元素的位置重排。

那么该前提条件是否必然成立呢?答案是必然成立。

假设原排序数组中存在需要调整的点 和点 ,且

为了让数组满足条件,它们都进行了“减少”操作的调整,分别变为了 ,如果触发位置重排的话,必然有

此时,我们能够通过调整它们的变化关系:点 变为点 、点 变成点 来确保同样满足条件,且不触发元素在有序数组中的位置重排。

alt

贪心

排序,限定第一位值为 ,从前往后处理,根据每一位是否「必须修改(与上一位差值是否大于 )」做决策,如果必须被修改,则修改为与前一值差值为 的较大数。

代码:

class Solution {
    public int maximumElementAfterDecrementingAndRearranging(int[] arr) {
        int n = arr.length;
        Arrays.sort(arr);
        arr[0] = 1;
        for (int i = 1; i < n; i++) {
            if (arr[i] - arr[i - 1] > 1) {
                arr[i] = arr[i - 1] + 1;
            }
        }
        return arr[n - 1];
    }
}
  • 时间复杂度:假定 Arrays.sort 使用的是双轴快排实现。复杂度为
  • 空间复杂度:假定 Arrays.sort 使用的是双轴快排实现。复杂度为

最后

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

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

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

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

更多更全更热门的「笔试/面试」相关资料可访问排版精美的 合集新基地 🎉🎉

本文由 mdnice 多平台发布

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

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

相关文章

Cesium.Entity图片纹理在不同观察角度有不同亮度

Cesium.Entity图片纹理在不同观察角度有不同亮度 测试代码&#xff1a; viewer.entities.add({rectangle: {coordinates: Cesium.Rectangle.fromDegrees(-92.0, 30.0, -76.0, 40.0),material: "../images/rect.png",} }); 测试图片&#xff1a; rect.png 这个图片…

vue 学习笔记 简单实验

1.代码(html) <script src"https://unpkg.com/vuenext" rel"external nofollow" ></script> <div id"counter">Counter: {{ counter }} </div> <script> const Counter {data() {return {counter: 5}} } Vue.cr…

★80交流驱动器通过rs485接口设置速度(附ascii表)

1抓取的数据及解析 2手册上的通信协议及数据帧说明 说明:双向传输项目 3硬件接线注意事项 用的RSJ45端子&#xff0c;双传项目中&#xff0c;一头用的pin6的水晶头子&#xff08;直流离心机上用过是可以的&#xff09;&#xff0c;另一个用的pin8的水晶头子&#xff0c;这里最…

Linux常用命令——dhcrelay命令

在线Linux命令查询工具 dhcrelay 使用dhcrelay命令可以提供中继DHCP和BOOTP请求 补充说明 dhcrelay命令使用dhcrelay命令可以提供中继DHCP和BOOTP请求&#xff0c;从一个没有DHCP服务器的子网直接连接到其它子网内的一个或多个DHCP服务器。该命令在DHCP中继服务器上使用&am…

【MPLS LDP】安全策略

正在备考华为认证的小伙伴应该知道&#xff0c;除了理论知识外&#xff0c;刷题也相当重要&#xff0c;周工这里有一份HCIAHCIP-Datacom带解析的最新题库 点赞留言 即可领取。 LDP协议 安全策略介绍 LDP MD5验证 MD5称为Message-Digest Algorithm 5&#xff0c;是国际标准摘…

面试题(一)

目录 一.@Transactional 原理和常见的坑 前期准备 事务不生效的几种 Case 原理 源码解读 @Transactional 执行机制 private 导致事务不生效原因 异常不匹配原因 两种 @Transactional Spring @Transactional javax.transaction.Transactional 二.@Autowired 和 @R…

EasyExcel实现文件分批次导入

文章目录 EasyExcel引入依赖表结构项目结构DozerUtils工具类实体类StudentController监听类ServiceServiceImplmapper 启动项目测试测试数据PostMan测试 EasyExcel EasyExcel是一个基于Java的、快速、简洁、解决大文件内存溢出的Excel处理工具。 他能让你在不用考虑性能、内存…

智能交通顶会 IEEE ITSC工作 LimSim:长期交互式多场景交通仿真器(已开源)

交通仿真器是自动驾驶技术发展的重要支撑。交通仿真器可以在虚拟环境中仿真各种交通场景和车辆行驶情况&#xff0c;从而提高测试效率、降低测试风险、提高测试准确性和加速开发周期&#xff0c;是自动驾驶技术验证和优化的重要手段之一。 **LimSim是由上海人工智能实验室智能…

怎么把PDF转成Word?需要注意什么事项?

PDF是一种常见的文档格式&#xff0c;但是与Word文档不同&#xff0c;PDF文件通常不能直接编辑。如果您想编辑PDF文件中的文本&#xff0c;或者想将PDF文件转换为Word文档&#xff0c;下面我们就来看一看把PDF转成Word有哪些方法和注意事项。 PDF转Word工具 有许多将PDF转换为…

AndroidStudio升级后总是Read Time Out的解决办法

AndroidStudio升级后在gradle的时候总是Time out&#xff0c;遇到过多次&#xff0c;总结一下解决办法 1、gradle下载超时 在工程目录../gradle/wrapper/gradle-wrapper.properties中找到gradle版本的下载链接&#xff0c;如下图&#xff1a; 将其复制到迅雷里下载&#xff0…

vue和react学哪一个比较有助于以后发展?

前言 首先声明vue和react这两个框架都是很优秀的前端框架&#xff0c;使用的人群下载量上数量也是相当的庞大&#xff0c;这篇文章没有贬低或者攻击任何一个框架的意思&#xff0c;只在于根据答主的问题来对这两个框架做出对比&#xff0c;以方便大家更加清晰的了解到当下vue和…

小迪和小捷的太空之旅——汽车篇

前情提要 书接上回&#xff0c;小迪与小捷接到外星人的委托&#xff0c;前往其母星拿取特殊小硬件。 在这个陌生的星球&#xff0c;小迪和小捷将遇到怎样的故事呢&#xff1f;

flask实现获取到上传的文件--postman实战

python&#xff1a; index_page.route("/upload",methods["POST"]) def upload():f request.files[file]return "request:%s,params:%s,var_a:%s" % (request.method, request.files, f)postman测试 也可以使用curl进行测试&#xff0c;post可…

高防护等级工业RFID读写器

工业环境恶劣&#xff0c;RFID工业读写器要能够在工业领域应用必须满足一定的防护等级&#xff0c;才能避免外界灰尘油污对设备产生影响&#xff0c;因此企业选择一款高防护等级的读写器尤为重要。下面本文就为大家介绍一下工业读写器对应的防护等级&#xff0c;给大家一个参考…

Wireshark数据抓包分析之互联网控制报文协议_ICMP

一、实验目的: 通过使用wireshark抓取的ICMP数据包对这个ICMP控制报文进行分析 二、预备知识&#xff1a; 1.ICMP协议概述&#xff1a;ICMP是Internet Control Message Protocol的缩写&#xff0c;即互联网控制报文协议。它是TCP/IP协议族的一个子协议&#xff0c;用于IP主机、…

c++ qt--页面布局(第五部分)

c qt–页面布局&#xff08;第五部分&#xff09; 一.页面布局 在设计页面的左侧一栏的组件中我们可以看到进行页面布局的一些组件 布局组件的使用 1.水平布局 使用&#xff1a;将别的组件拖到水平布局的组件中即可&#xff0c;可以选择是在哪个位置 2.垂直布局 使用&…

线性代数的学习和整理5: 矩阵的加减乘除及其几何意义

目录 1 矩阵加法 1.1 矩阵加法的定义 1.2 加法的属性 1.2.1 只有同类型&#xff0c;相同n*m的矩阵才可以相加 1.2.1 矩阵加法的可交换律&#xff1a; 1.2.2 矩阵加法的可结合律&#xff1a; 1.3矩阵加法的几何意义 2 矩阵的减法 2.1 矩阵减法定义和原理基本同 矩阵的…

MongoDB【CRUD练习-条件查询-文档关系】

练习1-CRUD // 进入test数据库 use test; // 查询文档内容 db.students.find(); // 显示当前数据库中所有集合 show collections; // 向数据库的user集合中插入一个文档 db.users.insertOne({username: "lyh"} ); // 查看当前数据库中所有的集合 发现users集合被创建…

JavaFX笔记

设置透明 group.setOpacity(0.5);&#xff0c;这里给group设置了半透明&#xff0c;其中的组件也会跟着变化。 按钮的字体 按钮背景 麻烦方法 paint是颜色 CornerRadii是圆角的弧度 CSS方法 单击事件

java八股文面试[java基础]——反射

知识来源&#xff1a; 【23版面试突击】讲讲JAVA的反射机制_哔哩哔哩_bilibili