Codeforces Round #688 (Div. 2)B. Suffix Operations(思维+贡献)

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+1ai)+abs(ai+2ai+1)...abs(anan1),将 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+2ai)其他的保持不变,原本将 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+1ai)+abs(ai+2ai+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();
	}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值