B. Suffix Operations
题意
给你n个数,然后有两个规则:
- 给后缀增1
- 给后缀减1
其中有一个特殊操作,可以任意修改一个数字。
问你使n个数变为相同的最小操作书是多少。
思路
考虑不加特殊操作的使 a i + 1 a_{i+1} ai+1变为 a i a_i ai的最小代价, a n s = a b s ( a i + 1 − a i ) + a b s ( a i + 2 − a i + 1 ) . . . a b s ( a n − a n − 1 ) ans = abs(a_{i + 1} - a_i) + abs(a_{i+ 2}- a{i +1})...abs(a_{n} - a_{n - 1}) ans=abs(ai+1−ai)+abs(ai+2−ai+1)...abs(an−an−1),将 a i + 1 a_{i+1} ai+1变为 a i a_i ai后,要使 a i + 1 a_{i+1} ai+1之后的也变为 a i a_i ai只需要将其变为前一个数,因为整个过程是相对的。
考虑加入特殊操作,将任意一个数变为变一个数。
这里为什么是要变为前一个数呢?考虑变一个其他位置的数变为任意数,那么可能在原基础需要增加操作,但是变为前一个数一定是减小操作。所以这里是变为前一个数。
加入特殊操作后考虑将 a i + 1 a_{i+1} ai+1变为 a i a_i ai,然后考虑其他位变化 a i + 2 a_{i+2} ai+2变为 a b s ( a i + 2 − a i ) abs(a_{i + 2} - a{i}) abs(ai+2−ai)其他的保持不变,原本将 a i + 1 , a i + 2 a_{i+1},a_{i+2} ai+1,ai+2变为 a i a_i ai需要的代价是 a b s ( a i + 1 − a i ) + a b s ( a i + 2 − a i + 1 ) abs(a_{i + 1} - a_i) + abs(a_{i + 2} - a_{i + 1}) abs(ai+1−ai)+abs(ai+2−ai+1),二者差值的最大值就是可以省的最大值。
需要特殊考虑第一位和最后一位
#include<bits/stdc++.h>
using namespace std;
const int N = 2e5 + 100;
typedef long long LL;
//#define int long long
int a[N];
void solve() {
int n; cin >> n;
for (int i = 1; i <= n; ++i) cin >> a[i];
LL ans = 0;
for (int i = 2; i <= n; ++i) ans += abs(a[i] - a[i - 1]);
int ma = 0;
for (int i = 2; i < n; ++i) {
ma = max(ma, abs(a[i + 1] - a[i]) + abs(a[i] - a[i - 1]) - abs(a[i + 1] - a[i - 1]));
}
ma = max(ma, abs(a[2] - a[1]));
ma = max(ma, abs(a[n] - a[n - 1]));
cout << ans - ma << endl;
}
signed main() {
ios::sync_with_stdio(0);
cin.tie(0);
int T = 1;
cin >> T;
while (T--) {
solve();
}
}