Leetcoder Day29| 贪心算法part03

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.mfbz.cn/a/416568.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【数据结构】双链表解析+完整代码(创建、插入、删除)

3.2 双链表 3.2.1 双链表的定义 定义 单链表的缺点&#xff1a;无法逆向操作&#xff0c;插入删除时只能从头开始遍历&#xff0c;很不方便。 双链表&#xff1a;每个结点都定义两个指针prior和next&#xff0c;分别指向前驱和后继&#xff0c;可进可退。 typedef struct DNod…

STM32F103学习笔记(七) PWR电源管理(原理篇)

目录 1. PWR电源管理简介 2. STM32F103的PWR模块概述 2.1 PWR模块的基本工作原理 2.2 电源管理的功能和特点 3. PWR模块的常见应用场景 4. 常见问题与解决方案 1. PWR电源管理简介 PWR&#xff08;Power&#xff09;模块是STM32F103系列微控制器中的一个重要组成部分&…

阿里云短信验证笔记

1.了解阿里云的权限操作 进入AccessKey管理 选择子用户 创建用户组和用户 先创建用户组&#xff0c;建好再进行权限分配 添加短信管理权限 创建用户 创建好后的id和密码在此处下载可以得到 2.开通阿里云短信服务 进行申请&#xff0c;配置短信模板 阿里云短信API文档 短信服务…

xss过waf的小姿势

今天看大佬的视频学到了几个操作 首先是拆分发可以用self将被过滤的函数进行拆分 如下图我用self将alert拆分成两段依然成功执行 然后学习另一种姿势 <svg id"YWxlcnQoIlhTUyIp"><img src1 οnerrοr"window[eval](atob(document.getElementsByTagNa…

zk和etcd的读一致性对比

背景 zk和etcd都是日常我们用到的分布式一致性的组件集群&#xff0c;不过他们在读一致性上还是有一些差别的&#xff0c;本文就来对比一下 zk和etcd的读一致性对比 如果读客户端没有通过zk或者etcd自带的watcher监听的方式监听某个写客户端写入的内容&#xff0c;而是依赖写…

Spring Task的应用

介绍 Spring Task是Spring框架提供的任务调度工具&#xff0c;可以按照约定的时间自动执行某个代码逻辑。 定位&#xff1a; 定时任务框架 作用&#xff1a; 定时自动执行某段Java代码 应用场景&#xff1a; 引用卡每月还款提醒、银行贷款每月还款提醒、火车票售票系统处理未支…

力扣每日一题 统计可能的树根数目 换根DP

Problem: 2581. 统计可能的树根数目 文章目录 思路复杂度Code 思路 &#x1f468;‍&#x1f3eb; 灵神 复杂度 时间复杂度: O ( n m ) O(nm) O(nm) 空间复杂度: O ( n m ) O(nm) O(nm) Code class Solution {List<Integer>[] g;// 邻接表Set<Long> se…

安泰电压放大器怎么用

电压放大器是一种常见的电子设备&#xff0c;用于放大电压信号。它是许多电子系统中的关键组件之一&#xff0c;可以增加电压信号的幅度&#xff0c;以便于后续的处理和驱动其他设备。以下是电压放大器的使用方法&#xff1a; 确定需求&#xff1a;首先&#xff0c;明确你的需求…

Jquery操作DOM对象

文章目录 目录 文章目录 本章目标 一.DOM操作分类 二.JQuery中的DOM操作 内容操作 属性值操作 节点操作 节点属性操作 节点遍历 总结 本章目标 使用Jquery操作网页元素使用JQuery操作文本与属性值内容使用JQuery操作DOM节点使用Jquery遍历DOM节点使用JQuery操作CSS-DOM 一…

linux下 将指定网卡名加入udp组播代码示例(端口复用)

linux下可通过指定网卡名字来将网卡加入组播 一般来说网卡名为 eth0,eth1之类的,具体的需要通过ifconfig查看 具体逻辑就是通过指定的网卡名来获取网卡ip,后面就跟普通的创建udp套接字一样了,需要注意的是将网卡绑定到组播最好启用地址重用来避免端口相同导致的失败 SO_REUSE…

Python程序打包成exe可执行文件的常用方法

在Python中,您可以使用一些工具将您的Python程序打包成可执行文件(.exe)。以下是一些常用的工具: PyInstaller: PyInstaller是一个流行的工具,它可以将Python脚本打包成独立的可执行文件,支持Windows、Linux和Mac。您可以使用以下命令安装PyInstaller: pip install pyin…

疾控污水采样设备需具备云控功能吗

疾控污水采样设备是否需要具备云控功能&#xff0c;是一个值得深入探讨的问题。从当前的技术发展趋势和实际应用需求来看&#xff0c;具备云控功能的疾控污水采样设备具有显著的优势和必要性。 第一&#xff0c;云控技术的应用可以实现远程监控和管理。在污水采样过程中&#…

测试环境搭建整套大数据系统(七:集群搭建kafka(2.13)+flink(1.13.6)+dinky+hudi)

一&#xff1a;搭建kafka。 1. 三台机器执行以下命令。 cd /opt wget wget https://dlcdn.apache.org/kafka/3.6.1/kafka_2.13-3.6.1.tgz tar zxvf kafka_2.13-3.6.1.tgz cd kafka_2.13-3.6.1/config vim server.properties修改以下俩内容 1.三台机器分别给予各自的broker_id…

在Ubuntu中安装Anaconda和创建虚拟环境(保姆级教学,值得借鉴与信任)

一、下载linux版本的Anaconda 1.方法一&#xff1a;官网下载 https://www.anaconda.com/download#downloads2.方法二&#xff1a;在清华大学开源软件镜像站里下载 https://mirrors.tuna.tsinghua.edu.cn/anaconda/archive/最后选择了Anaconda3-2022.10-Linux-x86_64.sh进行下…

AUTOMATION 自动化控制

Ansible介绍: 部署ansible:yum -y install ansible 批量管理服务器的工具2015年被红帽公司收购使用Python语言编写的基于ssh进行管理&#xff0c;所以不需要在被管端安装任何软件 ansible在管理远程主机的时候&#xff0c;主要是通过各种模块进行操作的 配置ansible管理环境: …

使用js写一个登录验证码效果

面试题 登录页面获取验证码的功能&#xff0c;用户点击获取验证码按钮(id”btn1”)&#xff0c;按文字变为“(N)后获取验证码”&#xff0c;N为倒计对秒数&#xff0c;从 60 开始&#xff0c;每秒减一&#xff0c;减到 0的时候&#xff0c;按钮文字变为“获取验证码”&#xff…

Python爬虫项目实战案例-批量下载网易云榜单音乐保存至本地

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua&#xff0c;在这里我会分享我的知识和经验。&#x…

Linux系统Docker部署StackEdit Markdown并实现公网访问本地编辑器

文章目录 前言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 测试…

国际伦敦银走势如此多变?如何应对

国际伦敦银走势由于受众多因素影响&#xff0c;因此它的多变是正常的&#xff0c;关键在于我们如何在多变的市场中应付自如&#xff0c;不光不会亏损&#xff0c;而且还要在多变的走势中抓住它的转折&#xff0c;找到入场的机会&#xff0c;下面我们就来讨论一下应对多变的国际…

刚入编的小学老师工资多少

“刚入编的小学老师&#xff0c;一个月能挣多少钱&#xff1f;”这个问题似乎总能激起一层波澜。不少人猜测&#xff0c;教师这份职业稳定、有寒暑假&#xff0c;工资应该不低吧&#xff1f;可真相往往出乎人们的意料。 事实上&#xff0c;一位刚入编的小学老师&#xff0c;月薪…
最新文章