AtCoder Regular Contest 072: C - Sequence

,

http://arc072.contest.atcoder.jp/tasks/arc072_a

solution

貪欲っぽい。$O(N)$。 $a_0$の移動先を$a_0, +1, -1$の$3$つから全て試す。 $a_n$まで決定したとき$a_{n+1}$は総和が条件を満たすような最小量だけ動かせばよい。 つまり総和はそのままか$+1, -1$になる。

これで十分なのは、$i \lt j$な$a_i$まで見て制約違反がなく$a_j$で違反するとき、$a_i$を動かして解消するのと$a_j$を動かして解消するのはどちらを動かしても移動量は同じであるため。

implementation

#include <cstdio>
#include <vector>
#include <algorithm>
#define repeat(i,n) for (int i = 0; (i) < int(n); ++(i))
using ll = long long;
using namespace std;
template <class T> inline void setmin(T & a, T const & b) { a = min(a, b); }
constexpr ll inf = ll(1e18)+9;

ll solve(vector<ll> const & a, ll initial) {
    int n = a.size();
    ll cnt = llabs(initial);
    ll acc = a[0] + initial;
    repeat (i,n-1) {
        ll nacc = acc + a[i + 1];
        if (acc > 0 and nacc >= 0) {
            cnt += nacc + 1;
            nacc = -1;
        }
        if (acc < 0 and nacc <= 0) {
            cnt += - nacc + 1;
            nacc = 1;
        }
        acc = nacc;
    }
    return cnt;
}

int main() {
    int n; scanf("%d", &n);
    vector<ll> a(n); repeat (i,n) scanf("%lld", &a[i]);
    ll result = inf;
    if (a[0] != 0) setmin(result, solve(a, 0));
    if (a[0] >= 0) setmin(result, solve(a, - a[0] - 1));
    if (a[0] <= 0) setmin(result, solve(a, - a[0] + 1));
    printf("%lld\n", result);
    return 0;
}