2019牛客暑期多校訓練營(第二場) - J - Go on Strike! - 前綴和預處理


題目鏈接:https://ac.nowcoder.com/acm/contest/882/C
來自:山東大學FST_stay_night的的題解,加入一些注釋幫助理解神仙代碼。

好像題解被套了一次又一次

要學習的地方我覺得是2點:

1.使用dp(貪心)的思想求出每段所在的連續段
2.因為前綴和是連續變化的,可以用lazy標記來代替樹狀數組來維護。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;

#define ERR(args...) { string _s = #args; replace(_s.begin(), _s.end(), ',', ' '); stringstream _ss(_s); istream_iterator<string> _it(_ss); err(_it, args); }

void err(istream_iterator<string> it) {
    cerr << "\n";
}
template<typename T, typename... Args>
void err(istream_iterator<string> it, T a, Args... args) {
    cerr << *it << "=" << a << ", ";
    err(++it, args...);
}

#define ERR1(arg,n) { cerr<<""<<#arg<<"=\n  "; for(int i=1;i<=n;i++) cerr<<arg[i]<<" "; cerr<<"\n"; }
#define ERR2(arg,n,m) { cerr<<""<<#arg<<"=\n"; for(int i=1;i<=n;i++) { cerr<<"  "; for(int j=1;j<=m;j++)cerr<<arg[i][j]<<" "; cerr<<"\n"; } }

const int INF = 0x3f3f3f3f;
const int MAXN = 10000000, MAXM = 1000000;

int l[MAXM + 5], r[MAXM + 5], f[MAXM + 5], g[MAXM + 5];
int sum[MAXN * 3 + 5], b[MAXN * 3 + 5], c[MAXN * 3 + 5];

int main() {
#ifdef Yinku
    freopen("Yinku.in", "r", stdin);
#endif // Yinku
    int n;
    scanf("%d", &n);
    for(int i = 1; i <= n; i++)
        scanf("%d%d", &l[i], &r[i]);
    f[1] = r[1] - l[1] + 1;
    //f[i]以i段右端點為結尾的能構造出的最大的前綴和
    for(int i = 2; i <= n; i++)
        f[i] = max(0, f[i - 1] - (l[i] - r[i - 1] - 1)) + r[i] - l[i] + 1;
    //0:以i-1段右端點結尾的能構造出的最大的前綴和都不足夠跨過[i-1,i]之間的-1
    //f[i - 1] - (l[i] - r[i - 1] - 1):跨過之后還剩下多少貢獻給這段
    g[n] = r[n] - l[n] + 1;
    //g[i]以i段左端點為開頭的能構造出的最大的前綴和
    for(int i = n - 1; i >= 1; i--)
        g[i] = max(0, g[i + 1] - (l[i + 1] - r[i] - 1)) + r[i] - l[i] + 1;
    //ERR1(f, n);
    //ERR1(g, n);
    int i = 1, base = 10000000;
    ll ans = 0;
    while(i <= n) {
        int j = i + 1;
        while(j <= n && g[j] + f[j - 1] >= l[j] - r[j - 1] - 1) {
            //說明這個[j-1,j]之間的-1段可以因為兩側的f[j-1]和g[j]足夠大而連接起來
            j++;
        }
        j--;
        //此時j是從i開始最遠能夠連接到的區間
        int left = max(0, l[i] - g[i]), right = min(1000000000 - 1, r[j] + f[j]);
        //left,right是至少會產生一個貢獻的范圍
        //ERR(left, right);
        int t = i, mi = INF, mx = 0;
        sum[0] = 0;
        for(int k = left; k <= right; k++) {
            //統計這一整段可連接區間的前綴和
            if(k >= l[t] && k <= r[t])
                sum[k - left + 1] = sum[k - left] + 1;
            else
                sum[k - left + 1] = sum[k - left] - 1;
            if(k == r[t])
                t++;
            mi = min(mi, sum[k - left + 1] + base);
            mx = max(mx, sum[k - left + 1] + base);
            //b記錄前綴和出現過的次數
            b[sum[k - left + 1] + base] ++;
        }
        //ERR1(sum, right);
        //b記錄前綴和出現過的次數的后綴和
        for(int k = mx - 1; k >= mi; k--)
            b[k] += b[k + 1];
        //包含最左側點的貢獻
        ans += b[base + 1];
        for(int k = left; k <= right; k++) {
            t = sum[k - left + 1] + base;
            //t表示k位置sum的值
            //b[t+1]比t大的值的個數
            //c[t+1]比在k位置左側的比t大的值的個數的lazy
            b[t + 1] -= c[t + 1]; //把lazy加上去
            c[t] += c[t + 1] + 1; //lazy標記下移
            c[t + 1] = 0; //清空lazy
            ans += b[t + 1];
        }
        for(int k = mi; k <= mx; k++)
            b[k] = 0, c[k] = 0;
        i = j + 1;
    }
    printf("%lld", ans);
    return 0;
}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM