leetcode每日一题:1005. K 次取反后最大化的数组和

简介: leetcode每日一题:1005. K 次取反后最大化的数组和

题目

给你一个整数数组 nums 和一个整数 k ,按以下方法修改该数组:

选择某个下标 i 并将 nums[i] 替换为 -nums[i] 。

重复这个过程恰好 k 次。可以多次选择同一个下标 i 。

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


示例 1:

输入:nums = [4,2,3], k = 1

输出:5

解释:选择下标 1 ,nums 变为 [4,-2,3] 。


示例 2:

输入:nums = [3,-1,0,2], k = 3

输出:6

解释:选择下标 (1, 2, 2) ,nums 变为 [3,1,0,2] 。


示例 3:

输入:nums = [2,-3,-1,5,-4], k = 2

输出:13

解释:选择下标 (1, 4) ,nums 变为 [2,3,-1,5,4]


约束条件:

1 <= nums.length <= 104

-100 <= nums[i] <= 100

1 <= k <= 104


思路:

分析:通过题目和例题我们了解到本题是通过规定次数内的反转操作来使所求的数组和最大。

思路:本题可以直接写或者贪心法解决。

直接法:通过排序,从小到大排序数据,如果k的数够大,将负数都转为正数的话,那么进行二次排序从小到大,对剩余的k进行取余操作,只对最小的那个数进行翻转。下面看具体代码实现

class Solution {
    public int largestSumAfterKNegations(int[] nums, int k) {
        // 常规方法
        Arrays.sort(nums);
        int count = 0;
        for(int i =0;i<nums.length;i++){
            if(k>0 && nums[i]<0){
                nums[i] = -nums[i];
                k--;
            }
            count +=nums[i];
        }
        //二次排序
        Arrays.sort(nums);
        if(k>0){
        //这一块有点难理解
        //因为上面已经求了总和,现在看k是否是偶数,偶数的话就原封不动,奇数的话减去2倍值才能等价于 加上对它进行取反操作求和 
            return count-(k%2==0?0:2*nums[0]);
        }
        return count;
     }
}


贪心法:每走一步都满足最大,最后可以达到最大和,所以我们需要对数组中的数进行排序,按照绝对值大小进行逆向排序,java中自带方法如下:进行排序

 nums = IntStream.of(nums)
            .boxed()
            .sorted((o1,o2) -> Math.abs(o2)-Math.abs(o1))
            .mapToInt(Integer::intValue).toArray();


同时部分思路也有部分如同第一种方法,具体代码实现如下:

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();
        int num = 0;
        for(int i =0;i<nums.length;i++){
            if(nums[i]<0 && k>0){
                nums[i] = -nums[i];
                k--;
            }
            num+=nums[i];
        }
        if(k>0){
           return num-(k%2==0?0:2*nums[nums.length-1]);
        }
        return num;
    }
}

感谢您的阅读,希望对您有所帮助。关注我,完成每日算法自律打卡,什么时候开始都不晚!!

相关文章
|
5月前
|
Go
【LeetCode 热题100】DP 实战进阶:最长递增子序列、乘积最大子数组、分割等和子集(力扣300 / 152/ 416 )(Go语言版)
本文深入解析三道经典的动态规划问题:**最长递增子序列(LIS)**、**乘积最大子数组** 和 **分割等和子集**。 - **300. LIS** 通过 `dp[i]` 表示以第 `i` 个元素结尾的最长递增子序列长度,支持 O(n²) 动态规划与 O(n log n) 的二分优化。 - **152. 乘积最大子数组** 利用正负数特性,同时维护最大值与最小值的状态转移方程。 - **416. 分割等和子集** 转化为 0-1 背包问题,通过布尔型 DP 实现子集和判断。 总结对比了三题的状态定义与解法技巧,并延伸至相关变种问题,助你掌握动态规划的核心思想与灵活应用!
205 1
|
算法
Leetcode 初级算法 --- 数组篇
Leetcode 初级算法 --- 数组篇
131 0
LeetCode第53题最大子数组和
LeetCode第53题"最大子数组和"的解题方法,利用动态规划思想,通过一次遍历数组,维护到当前元素为止的最大子数组和,有效避免了复杂度更高的暴力解法。
LeetCode第53题最大子数组和
|
存储 Java API
LeetCode------合并两个有序数组(4)【数组】
这篇文章介绍了LeetCode上的"合并两个有序数组"问题,并提供了三种解法:第一种是使用Java的Arrays.sort()方法直接对合并后的数组进行排序;第二种是使用辅助数组和双指针技术进行合并;第三种则是从后向前的双指针方法,避免了使用额外的辅助数组。
LeetCode------合并两个有序数组(4)【数组】
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
341 2
LeetCode------找到所有数组中消失的数字(6)【数组】
这篇文章介绍了LeetCode上的"找到所有数组中消失的数字"问题,提供了一种解法,通过两次遍历来找出所有未在数组中出现的数字:第一次遍历将数组中的每个数字对应位置的值增加数组长度,第二次遍历找出所有未被增加的数字,即缺失的数字。
|
前端开发
LeetCode------移动零(5)【数组】
这篇文章介绍了LeetCode上的"移动零"问题,提出了一种使用双指针的原地操作解法,该方法首先将非零元素移动到数组前端并保持相对顺序,然后填充后续位置为零,以达到题目要求。
【LeetCode-每日一题】 删除排序数组中的重复项
【LeetCode-每日一题】 删除排序数组中的重复项
114 4
|
索引
Leetcode第三十三题(搜索旋转排序数组)
这篇文章介绍了解决LeetCode第33题“搜索旋转排序数组”的方法,该问题要求在旋转过的升序数组中找到给定目标值的索引,如果存在则返回索引,否则返回-1,文章提供了一个时间复杂度为O(logn)的二分搜索算法实现。
111 0
Leetcode第三十三题(搜索旋转排序数组)
LeetCode第81题搜索旋转排序数组 II
文章讲解了LeetCode第81题"搜索旋转排序数组 II"的解法,通过二分查找算法并加入去重逻辑来解决在旋转且含有重复元素的数组中搜索特定值的问题。
LeetCode第81题搜索旋转排序数组 II

热门文章

最新文章