其他分享
首页 > 其他分享> > 《河南省赛C》

《河南省赛C》

作者:互联网

https://ac.nowcoder.com/acm/contest/17148/C。

这题挺好的。

其实一开始已经差不多想到了。

前缀和+容斥。

但是这个容斥一直没想到怎么做,就硬冲线段树去了。

这里其实有一个很重要的信息。

假设L点的右边有解的位置是r。

那么对于所有L左边的点,他们的有解位置肯定<=r。

那么我们查询的时候。

对于区间[L,r]:先找到r的左有解位置pos.

那么对于[L,pos]里的点,他们的右有解位置肯定<=r。

那么我们把以他们为开头~n里的所有值加上(这里用前缀和来处理即可),然后再 - (pos - L + 1) * (n - r)即可。

因为这个信息的存在,我们减去的区间里的右有解位置都肯定<=r。不会存在减去负的情况,也满足正确性。

对于sum[i]表示以i ~ n里面满足条件的序列数。

我们先尺取维护出L[i] - i的最大左有解点,r[i] - i的最小有解点。

那么sum[i]就可以走一遍前缀和取出来。

这里还有个小坑。

如果下面异或处理的L和赋值的一样,那么在第一个L异或之后,L就变了。

所以要用不同的变量。

// Author: levil
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
typedef pair<LL,LL> pii;
const int N = 1e6 + 5;
const int M = 1e5 + 5;
const LL Mod = 998244353;
#define pi acos(-1)
#define INF 1e18
#define dbg(ax) cout << "now this num is " << ax << endl;
namespace FASTIO{
    inline LL read(){
        LL x = 0,f = 1;char c = getchar();
        while(c < '0' || c > '9'){if(c == '-') f = -1;c = getchar();}
        while(c >= '0' && c <= '9'){x = (x<<1)+(x<<3)+(c^48);c = getchar();}
        return x*f;
    }
}
using namespace FASTIO;

int n,m,k,a[N],lp[N],rp[N];
LL sum[N];
deque<int> Q;
map<int,int> mp;
LL solve(int L,int r) {
    int pos = lp[r];
    if(pos < L) return 0;
    LL ma = sum[L] - sum[pos + 1];
    return ma - 1LL * (pos - L + 1) * (n - r);
}
int main() {
    n = read(),m = read(),k = read();
    for(int i = 1;i <= n;++i) a[i] = read();
    int now = 0,cnt = 0;
    while(now <= n) {
        if(cnt < k) {
            if(now == n) break;
            int pre = 0;
            if(Q.size() > 0) pre = Q.back();
            Q.push_back(++now);
            mp[a[now]]++;
            if(mp[a[now]] == 1) {
                cnt++;
                if(cnt == k && Q.size() > 0) {
                    rp[Q.front()] = Q.back();
                    lp[Q.back()] = Q.front();
                }
            }
            else if(Q.size() > 0){
                lp[Q.back()] = lp[pre];
            }
        }
        else {
            int x = Q.front();
            Q.pop_front();
            mp[a[x]]--;
            if(mp[a[x]] == 0) cnt--;
            if(cnt == k && Q.size() > 0) {
                rp[Q.front()] = Q.back();
                lp[Q.back()] = Q.front();
            }
        }
    }
    for(int i = 1;i <= n;++i) {
        if(a[i] == a[i - 1]) {
            rp[i] = rp[i - 1];
            lp[i] = lp[i - 1];
        }
    }
    for(int i = n;i >= 1;--i) {
        sum[i] = sum[i + 1];
        if(rp[i] != 0) sum[i] += 1LL * (n - rp[i] + 1);
    }
 //   for(int i = 1;i <= n;++i) printf("L[%d] is %d r[%d] is %d sum[%d] is %lld\n",i,lp[i],i,rp[i],i,sum[i]);
    LL ans = 0;
    while(m--) {
        int LL,rr;LL = read(),rr = read();
        int L = min(LL ^ ans,rr ^ ans) + 1;
        int r = max(LL ^ ans,rr ^ ans) + 1;
        ans = solve(L,r);
        printf("%lld\n",ans);
    }

    system("pause");
    return 0;
}
/*
10 5 3
1 2 3 4 5 6 7 8 9 10
1 5
2 6
3 7
1 5

*/
View Code

 

标签:河南省,sum,back,pos,int,lp,front
来源: https://www.cnblogs.com/zwjzwj/p/14828970.html