CF1421E Swedish Heroes

个人觉得挺妙的一个结论题。

首先不难发现每个数对答案的贡献是 (a_i) 或者 (-a_i)

那么系数的序列应该有两个性质:必须存在两个相同的系数相邻。设有 (p) 个减号,(q) 个加号,满足 (2p+q equiv 1 pmod{3})

证明后面再写。

那么知道这个性质,考虑记 (dp_{i,j,0/1,0/1}) 表示第 (i) 位,模 (3) 的余数,这一位的系数,是否存在两个相同的系数相邻的最大值。

时间复杂度 (O(n))

注意要考虑到 +-+-+-... 或者 -+-+-+... 这类情况是不合法的。

结论的证明:

对于 (n in [1,3]) 结论显然正确。

对于 (n geq 4),考虑合并之后 (p'=q+1,q'=p)

那么就有 (2(2p'+q')=3q+5)

显然 (3q+5 equiv 2 pmod{3}),所以 (2p+q equiv 1 pmod{3})

#include <bits/stdc++.h>
#define reg register
#define ll long long
#define ull unsigned long long
#define rep(i,a,n) for(reg int (i)=(a);(i)<=(n);++(i))
#define per(i,a,n) for(reg int (i)=(a);(i)>=(n);(i)--)
#define pb push_back
#define mp make_pair
#define fi first
#define se second
#define SZ(x) ((int) (x).size())
#define debug(typ...) fprintf(stderr, typ)
using namespace std;
const int iINF=numeric_limits<int>::max();
const ll lINF=numeric_limits<ll>::max();
int fastin() {
  reg int x = 0, ch = getchar(), f = 0;
  while(!isdigit(ch)) (ch == '-') && (f = 1), ch = getchar();
  while(isdigit(ch)) x = (x << 1) + (x << 3) + (ch ^ 48), ch = getchar();
  return f ? -x : x;
}
const int MAXN=2e5+10;
int n;
ll a[MAXN],dp[MAXN][3][2][2];
template<class typ>
void checkmax(reg typ &x,reg typ y) {
  if(x<y) x=y;
}
void work() {
  n=fastin();
  rep(i,1,n) a[i]=fastin();
  if(n==1) {
    printf("%lld
",a[1]);
    return;
  }
  memset(dp,-127,sizeof(dp));
  dp[1][2][0][0]=-a[1],dp[1][1][1][0]=a[1];
  rep(i,1,n-1) {
    rep(j,0,2) {
      checkmax(dp[i+1][(j+1)%3][1][1],max(dp[i][j][1][0],max(dp[i][j][1][1],dp[i][j][0][1]))+a[i+1]);
      checkmax(dp[i+1][(j+1)%3][1][0],dp[i][j][0][0]+a[i+1]);
      checkmax(dp[i+1][(j+2)%3][0][1],max(dp[i][j][0][0],max(dp[i][j][0][1],dp[i][j][1][1]))-a[i+1]);
      checkmax(dp[i+1][(j+2)%3][0][0],dp[i][j][1][0]-a[i+1]);
    }
  }
  printf("%lld
",max(dp[n][1][0][1],dp[n][1][1][1]));
}
signed main() {
  int _=1;
  // _=fastin();
  while(_--) work();
  return 0;
}
原文地址:https://www.cnblogs.com/Lonely-233/p/13899563.html