1005. K 次取反后最大化的数组和 - 力扣(LeetCode)

题目描述

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

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

题目示例

输入:A = [4,2,3], K = 1

输出:5

解释:选择索引 (1,) ,然后 A 变为 [4,-2,3]。

解题思路

使用贪心算法,两次贪心策略解决该题,首先将数组按照绝对值从大到小排序,然后进行以下两次贪心策略。

  • 第一次贪心,我们取反,首先取反绝对值最大的负数,也就是最小值。如果第一次贪心正好消耗完 k,就不需要第二次贪心了。
  • 第二次贪心,如果我们按照第一次贪心之后,还有取反次数,我们再将数组的最后一个元素(因为第一次贪心完成后,如果还有取反次数,代表数组所有的负数已经全被取反,数组只剩正数,数组最后一个元素即为最小值)进行取反(如果 k 还剩奇数的话,因为偶数的话来回取反还是原来的值,所以就不用取反了)

最后将取反完成的数组取总和返回结果。

参考代码

java 复制代码
class Solution {
    public int largestSumAfterKNegations(int[] nums, int k) {

        // 将数组按照绝对值大小排序
        nums = IntStream.of(nums)
		     .boxed()
		     .sorted((o1, o2) -> Math.abs(o2) - Math.abs(o1))
		     .mapToInt(Integer::intValue).toArray();
        // 第一次贪心,将最大的负数取反
        for(int i = 0; i < nums.length; i++) {
            if(nums[i] < 0 && k > 0) {
                nums[i] *= -1;
                k--;
            }
        }
        // 第二次贪心,如果还有取反机会,将最小数字取反
        if(k % 2 == 1) {
            nums[nums.length - 1] *= -1;
        }
        // 取修改完数组的总和
        return Arrays.stream(nums).sum();
    }
}
相关推荐
南宫萧幕1 分钟前
锂电池二阶 RC 模型仿真实战:从理论解析到 Simulink 闭环搭建全流程
开发语言·人工智能·算法·机器学习
故事和你912 分钟前
洛谷-数据结构2-1-二叉堆与树状数组2
开发语言·javascript·数据结构·算法·ecmascript·动态规划·图论
炸膛坦客5 分钟前
嵌入式 - 数据结构与算法:(1-8)数据结构 - 栈(Stack)
c语言·数据结构
智者知已应修善业5 分钟前
【51单片机流水灯中断嵌套,低优先级中断完成后如何返回主程序】2023-10-15
c++·经验分享·笔记·算法·51单片机
北顾笙9807 分钟前
day41-数据结构力扣
数据结构·算法·leetcode
凯瑟琳.奥古斯特8 分钟前
懒加载技巧优化栈增减操作(力扣3629)
开发语言·数据结构·算法
hans汉斯8 分钟前
基于LSTM与扩展卡尔曼滤波的无人机机载电子磁干扰补偿研究
开发语言·人工智能·算法·目标检测·lstm·人机交互·无人机
sheeta19982 小时前
LeetCode 每日一题笔记 日期:2026.05.08 题目:3629. 素数跳跃最小次数
笔记·算法·leetcode
叼烟扛炮2 小时前
C++ 知识点08 类与对象
开发语言·c++·算法·类和对象
米粒12 小时前
力扣算法刷题 Day 63 Bellman_ford 算法
数据库·算法·leetcode