这是悦乐书的第376次更新,第403篇原创
java
今天介绍的是LeetCode
算法题中Easy
级别的第237
题(顺位题号是1005
)。给定一个整数数组A
,咱们必须按如下方式修改数组:咱们选择一个i并用-A[i]
替换A[i]
,重复这个过程K
次。(咱们能够屡次选择相同的索引。)
以这种方式修改后,返回数组可能的最大总和。例如:算法
输入:A = [4,2,3], K = 1
输出:5
说明:选择索引(1,),A
变为[4,-2,3]
。数组
输入:A = [3,-1,0,2], K = 3
输出:6
说明:选择索引(1, 2, 2),A
变为[3,1,0,2]
。数据结构
输入:A = [2,-3,-1,5,-4], K = 2
输出:13
说明:选择索引(1, 4),A
变为[2,3,-1,5,4]
。优化
注意:code
1 <= A.length <= 10000排序
1 <= K <= 10000索引
-100 <= A[i] <= 100
io
题目的意思是将A
中的数进行取反(正变负,负变正)K
次,能够重复对一个元素取反,最后求A
中元素总和的最大值。取反能够分为两种状况:class
当A
中都是正数的时候,好比{1,2,4,6},若是K
是偶数,那么能够不用进行取反操做,由于负负得正;若是K
是奇数,则只须要对最小的数取反一次便可。
当A
中有正数也有负数的时候,好比{-4,-3,-1,2,5}
,此时对负数元素进行取反操做,直到当前元素大于0或者K次转换已用完,此时针对K
中剩余的转换次数,又能够细分为两种状况:
(1)K
中剩余的转换次数为偶数,即A中元素全是正数,依据负负得正,不用再进行额外的转换了。
(2)K
中剩余的转换次数为奇数,即还须要再将某个元素转换一次,而为了元素总和最大,须要比较当前元素(正数)和前一个元素(负数)的绝对值大小,对较小的元素进行取反。
最后使用一个for
循环,计算A
中全部元素总和。
public int largestSumAfterKNegations(int[] A, int K) { int sum = 0; Arrays.sort(A); if (A[0] >= 0) { if (K%2 != 0) { A[0] = -A[0]; } } else { int i = 0; while (A[i] <= 0 && K > 0) { A[i] = -A[i]; i++; K--; } // K中剩余有转换次数且为奇数 if (K > 0 && K%2 != 0) { // 取较小的元素进行取反 if (Math.abs(A[i]) < Math.abs(A[i-1])) { A[i] = -A[i]; } else { A[i-1] = -A[i-1]; } } } // 求和 for (int num : A) { sum += num; } return sum; }
思路和第一种解法相似,只是在处理K
中剩余的转换次数为奇数这个问题上作了下优化,在第一种解法中,咱们是比较最后一次转换的负数和它前一个正数的大小,换种角度理解,这两个数已经处于全部元素中的底部了,就像一个U
型曲线的底同样,而为了元素总和最大,须要将较小的值取反,即找出全部元素中的最小值便可。
先对A
排序,利用循环,将前面的负数进行转换,顺便计算元素总和并找出最小元素,若是A
中负数已所有转换完成(A
中已不包含负数),且K中还有剩余的转换次数,而且剩余的转换次数为奇数,则将最小的正数减去两次,由于sum
是所有元素的和,且所有元素都是正数,对最小的元素取反后变为负数,则sum须要减两次最小元素。例如{1,2,3}
的和为6
,对1
取反后变为{-1,2,3}
,其和为4
。
public int largestSumAfterKNegations2(int[] A, int K) { Arrays.sort(A); int sum = 0, min = Integer.MAX_VALUE; for (int num : A) { if (num <= 0 && K > 0) { num = -num; K--; } sum += num; min = Math.min(min, num); } // K中有剩余转换次数且为奇数 if (K > 0 && K%2 != 0) { // 减去2次最小值 sum -= min*2; } return sum; }
算法专题目前已连续日更超过七个月,算法题文章242+篇,公众号对话框回复【数据结构与算法】、【算法】、【数据结构】中的任一关键词,获取系列文章合集。
以上就是所有内容,若是你们有什么好的解法思路、建议或者其余问题,能够下方留言交流,点赞、留言、转发就是对我最大的回报和支持!