1. 程式人生 > >ARC096D Static Sushi 記錄前綴最大值

ARC096D Static Sushi 記錄前綴最大值

sum swap ont mes scan scanf amp 枚舉 ron

傳送門ARC096D

思路:枚舉 + 記錄前綴最大值

我們考慮先逆時針走。

枚舉逆時針的步數i,那麽我們先從0->n+1-i,然後回到起點n+1-i->0

然後順時針走,加上前綴最大值preMax[n-i]就好了。

先順時針走同理。

code

#include <iostream>
using namespace std;
typedef long long LL;
const int N = 100000+10;
const LL INF = 1e16;

int n; 
LL c, x[N], v[N];
LL w[N], preMax[N];

LL solve() {
    preMax[0] = -INF;
    for(int i=1;i<=n;i++) {
        w[i]=v[i]-(x[i]-x[i-1]);
        w[i]=w[i-1]+w[i];
        preMax[i]=max(preMax[i-1],w[i]);
    }
    LL ret = -INF;
    ret = preMax[n];

    LL sum = 0;
    for(int i=n;i>=1;i--) {
        sum += v[i];
        ret=max(ret, sum-2*(c-x[i])+preMax[i-1]);
    }
    return ret;
}
int main() {
    scanf("%d %lld",&n,&c);
    for(int i=1;i<=n;i++) 
        scanf("%lld %lld", &x[i], &v[i]);
    LL ans = max(solve(),0LL);

    for(int i=1;i<=n;i++)
        if(i<n+1-i) {
            swap(x[i],x[n+1-i]);
            swap(v[i],v[n+1-i]);
        }
    for(int i=1;i<=n;i++)
        x[i]=c-x[i];
    ans = max(ans,solve());
    cout<<ans<<endl;
}

ARC096D Static Sushi 記錄前綴最大值