I. Array Negations
题目链接-I. Array Negations
题目大意
给你一个含
n
n
n个元素的数组,你必须进行k个求反操作,即每次操作都从数组中选择一个元素
a
i
ai
ai,并将其替换为
−
a
i
−ai
−ai,请你输出在进行
k
k
k的取反操作之后,数组
a
a
a所有元素的最大和
解题思路
$贪心$4
- 先将所有元素排序,若负数数目大于 k k k,就把前 k k k个数取反,然后计算数组元素和输出即可
- 若负数数目小于等于 k k k,就把所有负数取反,然后判断所有负数取反后,剩余次数是奇数还是偶数
- 若为偶数,即可对任意一个数进行偶数次取反操作,这时数的大小不会发生改变,所以直接计算数组元素和输出即可
- 若为奇数,可将操作后数组排序,然后将第一个元素取反即可,此时可保证 k k k次操作后数组元素和最大
- 具体操作见代码
附上代码
#pragma GCC optimize("-Ofast","-funroll-all-loops")
#include<bits/stdc++.h>
#define int long long
#define lowbit(x) (x &(-x))
#define endl '\n'
using namespace std;
const int INF=0x3f3f3f3f;
const int dir[4][2]={-1,0,1,0,0,-1,0,1};
const double PI=acos(-1.0);
const double e=exp(1.0);
const double eps=1e-10;
const int M=1e9+7;
const int N=2e4+10;
typedef long long ll;
typedef pair<int,int> PII;
typedef unsigned long long ull;
int a[N];
signed main(){
ios::sync_with_stdio(false);
cin.tie(0);cout.tie(0);
int t;
cin>>t;
while(t--){
int n,k,ans=0;
cin>>n>>k;
for(int i=0;i<n;i++)
cin>>a[i];
sort(a,a+n);
for(int i=0;i<n&&k;i++){
if(a[i]<0){
a[i]=-a[i];
k--;
}
else break;
}
if(k%2){
sort(a,a+n);
a[0]=-a[0];
}
for(int i=0;i<n;i++)
ans+=a[i];
cout<<ans<<endl;
}
return 0;
}