Leetcoder Day29| 贪心算法part03

news2025/2/25 3:22:24

1005.K次取反后最大化的数组和

给定一个整数数组 A,我们只能用以下方法修改该数组:我们选择某个索引 i 并将 A[i] 替换为 -A[i],然后总共重复这个过程 K 次。(我们可以多次选择同一个索引 i。)

以这种方式修改数组后,返回数组可能的最大和。

示例 1:

  • 输入:A = [4,2,3], K = 1
  • 输出:5
  • 解释:选择索引 (1,) ,然后 A 变为 [4,-2,3]。

示例 2:

  • 输入:A = [3,-1,0,2], K = 3
  • 输出:6
  • 解释:选择索引 (1, 2, 2) ,然后 A 变为 [3,1,0,2]

本题可以操作同一个索引多次,因此

如果当前数组元素均为正数,且K为奇数时,操作最小的那个,如果为偶数,则操作谁都无所谓。

如果当前数组元素存在负数,从绝对值最大的负数开始都取反为正数,若还有剩余,剩下的若为奇数,挑最小的来进行取反,若为偶数,则无所谓。

因此可以将数组先进行排序。

import java.util.Arrays;

class Solution {
    public int largestSumAfterKNegations(int[] nums, int k) {
        Arrays.sort(nums);
        int res = 0;
        int i = 0; //记录第一个非负数索引
        // 将负数取反,直到k用完或遇到第一个非负数
        while (k > 0 && i < nums.length && nums[i] < 0) {
            nums[i]= -nums[i];
            k--;
            i++;
        }
        Arrays.sort(nums);
        // 如果k为奇数且仍有剩余,则将最小的非负数取反
        if (k % 2 == 1 ) {
            nums[0]=-nums[0];
        }
        // 加上剩余的数字
        for(i=0; i<nums.length;i++){
            res += nums[i];
        }
        return res;
    }
}

134. 加油站

在一条环路上有 N 个加油站,其中第 i 个加油站有汽油 gas[i] 升。

你有一辆油箱容量无限的汽车,从第 i 个加油站开往第 i+1 个加油站需要消耗汽油 cost[i] 升。你从其中的一个加油站出发,开始时油箱为空。

如果你可以绕环路行驶一周,则返回出发时加油站的编号,否则返回 -1。

说明:

  • 如果题目有解,该答案即为唯一答案。
  • 输入数组均为非空数组,且长度相同。
  • 输入数组中的元素均为非负数。

示例 1: 输入:

  • gas = [1,2,3,4,5]
  • cost = [3,4,5,1,2]

输出: 3 解释:

  • 从 3 号加油站(索引为 3 处)出发,可获得 4 升汽油。此时油箱有 = 0 + 4 = 4 升汽油
  • 开往 4 号加油站,此时油箱有 4 - 1 + 5 = 8 升汽油
  • 开往 0 号加油站,此时油箱有 8 - 2 + 1 = 7 升汽油
  • 开往 1 号加油站,此时油箱有 7 - 3 + 2 = 6 升汽油
  • 开往 2 号加油站,此时油箱有 6 - 4 + 3 = 5 升汽油
  • 开往 3 号加油站,你需要消耗 5 升汽油,正好足够你返回到 3 号加油站。
  • 因此,3 可为起始索引。

本题答案有唯一解,关键是找到start,本题需要满足的条件为:

第一个站点(g[i]-cost[i])>=0 start=i;找出满足条件的

设置一个变量统计剩余油量rest,rest=g[i]-cost[i]+g[i+1],rest>cost[i+1]

一种暴力方法就是,找到满足条件的解的集合,值为索引,然后遍历,看最终谁满足这个条件。如果都不满足,说明没有解,返回-1;具体代码如下:
 

class Solution {
    /**
    第一个站点(g[i]-cost[i])>=0 start=i;找出满足条件的
    设置一个变量统计剩余油量rest,rest=g[i]-cost[i]+g[i+1],rest>cost[i+1]
     */
    public int canCompleteCircuit(int[] gas, int[] cost) {
        ArrayList<Integer> mayStart= new ArrayList<>();//定义一个数组存储可能的起点,值为加油站的索引
        int rest=0;
        //记录所有可能的起点
         for (int i = 0; i < gas.length; i++) {
            if (gas[i] - cost[i] >= 0) {
                mayStart.add(i);
            }
        }
        if(mayStart.size()==0) return -1;
        
        //每个可能的起点都走一遍,如果遇到rest<cost[i],就不返回
        for(int i=0;i<mayStart.size();i++){
            int start=mayStart.get(i); //获取当前起点   
            rest=0; //记得每次循环清空rest
            rest+=gas[start];
            int j=start;
            do{
                rest-=cost[j];
                if(rest<0) {//没有能够到达下一个点
                    break;

                } 
                rest+=gas[(j+1)%gas.length];
                j=(j+1)%gas.length;
            }while(j!=start);
            if(rest>=0) return start;
        }    
    return -1;
    }
}

但是事实证明,这种方法超出时间限制了。接下来就是按照代码随想录的思路:如果总油量大于等于总消耗,说明一定可以跑完。从0累加每个加油站的剩余油量gas[i]-cost[i],和为resSum,resSum一旦小于0,说明[0,1]区间内都不可能作为起点。因为如下图所示:

如果 curSum<0 说明 区间和1 + 区间和2 < 0, 那么 假设从上图中的位置开始计数curSum不会小于0的话,就是 区间和2>0。

区间和1 + 区间和2 < 0 同时 区间和2>0,只能说明区间和1 < 0, 那么就会从假设的箭头初就开始从新选择其实位置了。

那么局部最优:当前累加rest[i]的和curSum一旦小于0,起始位置至少要是i+1,因为从i之前开始一定不行。全局最优:找到可以跑一圈的起始位置

所以start就等于i+1开始,清空resSum重新计算。

class Solution {
    /**
    如果总油量大于等于总消耗,说明一定可以跑完。
    从0累加每个加油站的剩余油量gas[i]-cost[i],和为resSum,resSum一旦小于0,说明[0,1]区间内都不可能作为起点。
     */
    public int canCompleteCircuit(int[] gas, int[] cost) {
        int start=0;
        int resSum=0;
        int toSum=0;
        int toCon=0;
        for(int i=0;i<gas.length;i++){
            toSum+=gas[i];
            toCon+=cost[i];
            resSum+=gas[i]-cost[i];
            if(resSum<0){
                start=i+1;
                resSum=0;
            }
        }
        if(toCon>toSum) return -1;
        return start;
    }
}

135. 分发糖果

老师想给孩子们分发糖果,有 N 个孩子站成了一条直线,老师会根据每个孩子的表现,预先给他们评分。

你需要按照以下要求,帮助老师给这些孩子分发糖果:

  • 每个孩子至少分配到 1 个糖果。
  • 相邻的孩子中,评分高的孩子必须获得更多的糖果。

那么这样下来,老师至少需要准备多少颗糖果呢?

示例 1:

  • 输入: [1,0,2]
  • 输出: 5
  • 解释: 你可以分别给这三个孩子分发 2、1、2 颗糖果。

本题抽象成数学逻辑就是,当前孩子要跟左右孩子进行比较,判断他应该得到糖果的区间。可以确定比较顺序,一定要先确定一边,再确定另一边。

比如先确定左边,即判断当前孩子大于左孩子的情况,从前向后,将每个孩子跟左边的孩子进行比较,局部最优就是,只要评分比左边大ranking[i]>ranking[i-1],当前的孩子糖果就多一个eachCandy[i - 1] + 1。全局最优,相邻孩子中,评分高的右孩子都获得比左孩子更多的糖果。

然后确定右边,即判断当前孩子大于右孩子的情况,从后向前,如果大于右孩子ranking[i]>ranking[i+1]。,当前孩子就有两个选择,一个是eachCandy[i + 1] + 1(从右边这个加1得到的糖果数量),一个是candyVec[i](之前比较右孩子大于左孩子得到的糖果数量)。比较这俩个值谁大,就取谁大值。

最后遍历统计每个孩子的糖果数即可。

import java.lang.Math;
class Solution {
    public int candy(int[] ratings) {
        int candySum=0;
        int[] eachCandy =new int[ratings.length];
        for(int i=0;i<eachCandy.length;i++){
            eachCandy[i]=1;
        }
        //从前向后,跟左孩子进行比较,如果大于就将当前的糖果数+1
        for(int i=1;i<ratings.length;i++){
            if(ratings[i]>ratings[i-1]){
                eachCandy[i]=eachCandy[i-1]+1;
            }
        }
        //从后向前,跟左孩子进行比较,如果大于就将当前的糖果数+1
        for(int i=ratings.length-2;i>=0;i--){
            if(ratings[i]>ratings[i+1]){
                eachCandy[i]=Math.max(eachCandy[i], eachCandy[i+1]+1);
            }
        }
        for(int i=0;i<eachCandy.length;i++){
            candySum+=eachCandy[i];
        }
        return candySum;

    }
}

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

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

相关文章

Fast admin改变对话框的文字,并且绑定参数,确认和取消都可以做修改

需求&#xff1a;点击确认和拒绝都要做出对应的操作 列表添加一个buttos按钮用来单击触发对话框 {field: operate, title: __(Operate), table: table, events: Table.api.events.operate, buttons: [{name: click,title: __(点击执行事件),classname: btn btn-xs btn-info bt…

逆变器专题(6)-正负序分离(二阶广义积分器DSOGI)

相应仿真原件请移步资源下载 DSOGI作为一种常用的正负序分离方法&#xff0c;其可以在电压三相不平衡的状态下实现较为精准的锁相环。 原理 构建基于二阶广义积分器的自适应滤波器来实现90相角偏移和谐波的滤除。 其中&#xff0c;&#xff0c;表示原信号进行滞后90&#xff…

成功解决IndexError: Target 20 is out of bounds.

【PyTorch】成功解决IndexError: Target 20 is out of bounds. &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x1f448; 希望得到您…

Redisson限流算法

引入依赖 <dependency><groupId>org.redisson</groupId><artifactId>redisson-spring-boot-starter</artifactId><version>3.12.3</version> </dependency>建议版本使用3.15.5以上 使用 这边写了一个demo示例&#xff0c;定…

kswapd0挖矿病毒攻击记录

文章目录 一、起因与病毒分析1、起因2、阿里云告警2.1 恶意脚本代码执行12.2 恶意脚本代码执行22.3恶意脚本代码执行32.4 恶意脚本代码执行4 3、病毒简单分析3.1 病毒的初始化3.2 病毒本体执行 4、总结 二、ubuntu自救指南1、病毒清理2、如何防御 一、起因与病毒分析 1、起因 …

跳跃游戏Ⅱ

问题 给定一个长度为 n 的 0 索引整数数组 nums。初始位置为 nums[0]。 每个元素 nums[i] 表示从索引 i 向前跳转的最大长度。换句话说&#xff0c;如果你在 nums[i] 处&#xff0c;你可以跳转到任意 nums[i j] 处: 0 < j < nums[i] i j < n 返回到达 nums[n - …

【深蓝学院】移动机器人运动规划--第7章 集群机器人运动规划--笔记

文章目录 0. Contents1. Multi-Agent Path Finding (MAPF)1.1 HCA*1.2 Single-Agent A*1.3 ID1.4 M*1.5 Conflict-Based Search(CBS)1.6 ECBS1.6.1 heuristics1.6.2 Focal Search 2. Velocity Obstacle (VO&#xff0c;速度障碍物)2.1 VO2.2. RVO2.3 ORCA 3. Flocking model&am…

EAS web 界面加载后,隐藏按钮

效果&#xff1a;隐藏下列按钮&#xff1a; 实现方法&#xff1a; 1、创建数据装载事件&#xff1a; 2、隐藏按钮&#xff1a; afterOnloadHideEntryTBBBBBB:function(e){console.log("----------失败222&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&a…

Redis主从、哨兵、Redis Cluster集群架构

Redis主从、哨兵、Redis Cluster集群架构 Redis主从架构 Redis主从架构搭建 主从搭建的问题 如果同步数据失败&#xff0c;查看log日志报错无法连接&#xff0c;检查是否端口未开放出现”Error reply to PING from master:...“日志&#xff0c;修改参数protected-mode no …

WebDAV之π-Disk派盘 + DAVx⁵

DAVx⁵是一款通过标准 CardDAV 和 CalDAV 协议同步通讯录、日历的 Android 应用,支持 iCloud 等云服务,只需要在 Android 端安装,即可实现在 iPhone 与 Android 间双向同步通讯录、日历、提醒事项等数据。 资源自动检测,支持自签名证书,通过客户端证书进行身份验证;可以…

如何使用Windows系统电脑无公网ip远程桌面Ubuntu系统

文章目录 前言1. ubuntu安装VNC2. 设置vnc开机启动3. windows 安装VNC viewer连接工具4. 内网穿透4.1 安装cpolar【支持使用一键脚本命令安装】4.2 创建隧道映射4.3 测试公网远程访问 5. 配置固定TCP地址5.1 保留一个固定的公网TCP端口地址5.2 配置固定公网TCP端口地址5.3 测试…

SpringBoot整合rabbitmq-直连交换机队列(二)

说明&#xff1a;本文章主要是Direct定向/直连类型交换机的使用&#xff0c;它的大致流程是将一个队列绑定到一个直连交换机上&#xff0c;并赋予一个路由键 routingkey&#xff0c;当一个消息携带着路由值为routingkey&#xff0c;这个消息通过生产者发送给交换机时&#xff0…

Unity UI适配规则和对热门游戏适配策略的拆解

前言 本文会介绍一些关于UI适配的基础概念&#xff0c;并且统计了市面上常见的设备的分辨率的情况。同时通过拆解目前市面上较为成功的两款休闲游戏Royal Match和Monopoly GO(两款均为近期游戏付费榜前几的游戏)&#xff0c;大致推断出他们的适配策略&#xff0c;以供学习和参…

CP AutoSar之LIN Driver详细说明

本文遵循autosar标准&#xff1a;R22-11 1 简介 本文指定了 AUTOSAR 基础软件模块 LIN 驱动程序的功能、API 和配置。 1.1 范围 LIN驱动程序适用于ISO 17987主节点和从节点。AUTOSAR中的LIN实现偏离了本LIN驱动器规范中所述的ISO 17987规范&#xff0c;但LIN总线上的行为不…

Allure报告归纳与总结

一、环境准备&#xff1a; 提前准备环境:Java1.8 Allure2 解析过程: 1.安装 allure2(信赖Java1.8) allure官方下载地址&#xff1a; https://repo.maven.apache.org/maven2/io/qameta/allure/allure-commandline/ 2.安装 allure-pytest 命令:pip install allure-pytest 3.生成…

Mamba 作者谈 LLM 未来架构

文章目录 前言 1、为什么注意力机制有效&#xff1f; 2、注意力计算量呈平方级增长 3、Striped Hyena 是个什么模型&#xff1f; 4、什么是 Mamba? 5、Mamba 硬件优化 6、2024年架构预测 7、对 AI 更多的预测 本片文章来自【机器之心】对Mamba作者进行采访所进行的编译整理。 …

TCP缓存

TCP缓存是指TCP协议在数据传输过程中使用的一种机制&#xff0c;用于临时存储和管理数据包。它主要有三个作用&#xff1a;提高网络性能、保证数据的可靠性和实现流量控制。 首先&#xff0c;TCP缓存可以提高网络性能。当发送端发送数据时&#xff0c;TCP协议会将数据分割成若…

自动驾驶加速落地,激光雷达放量可期(上)

1 激光雷达应用广泛&#xff0c;汽车有望成最大催化 激光雷达&#xff08;LiDAR&#xff09;是一种主动遥感技术&#xff0c;通过测定传感器发出的激光在传感器与目标物体之间的传播距离&#xff0c;来分析目标地物表面的反射能量大小、反射波谱的幅度、频率和相位等信息&#…