zoukankan      html  css  js  c++  java
  • Codeforces348C

    Portal

    Description

    给出长度为(n(nleq10^5))的序列({a_n})以及(m(mleq10^5))个下标集合({S_m}(sum|S_i|leq10^5)),进行(q(qleq10^5))次操作:

    • 询问下标属于集合(S_k)的所有数之和。
    • 将下标属于集合(S_k)的所有数加(x)

    Solution

    (N_0=sqrt{sum|S_i|})
    我们把集合划分成轻集合与重集合,大小超过(N_0)的集合就是重集合。容易知道重集合的个数不超过(N_0)。对于每个重集合,记录sum表示该集合的和,add表示该集合总体被加了的值,cnt[i]表示该集合与集合(i)的交集大小。
    询问时,如果是重集合则输出sum;否则暴力求({a})上的和,再加上每个重集合对该轻集合的贡献add*cnt[k]
    修改时,如果是重集合则add+=x;否则暴力修改({a})。然后更新每个重集合的sum,也就是加上x*cnt[k]

    时间复杂度(O(qsqrt{sum|S_i|}))

    Code

    //Subset Sums
    #include <cstdio>
    #include <cmath>
    #include <cstring>
    #include <vector>
    #include <algorithm>
    using namespace std;
    typedef long long lint;
    inline int read()
    {
        int x=0,f=1; char ch=getchar();
        while(ch<'0'||'9'<ch) {if(ch=='-') f=-1; ch=getchar();}
        while('0'<=ch&&ch<='9') x=x*10+ch-'0',ch=getchar();
        return x*f;
    }
    int const N=1e5+1e3;
    int const N0=400;
    int n,m,q,n0; lint a[N];
    int ori[N];
    struct _set{int siz,id; vector<int> x;} s[N];
    bool cmpSiz(_set x,_set y) {return x.siz>y.siz;}
    bool tmp[N]; int cnt[N0][N]; lint add[N0],sum[N0];
    int main()
    {
        n=read(),m=read(),q=read();
        for(int i=1;i<=n;i++) a[i]=read();
        int sumK=0;
        for(int i=1;i<=m;i++)
        {
            s[i].siz=read(),s[i].id=i,s[i].x.push_back(0); sumK+=s[i].siz;
            for(int p=1;p<=s[i].siz;p++) s[i].x.push_back(read());
        }
        sort(s+1,s+m+1,cmpSiz);
        for(int i=1;i<=m;i++) ori[s[i].id]=i;
        n0=sqrt(sumK); int cnt1=0;
        for(int i=1;s[i].siz>=n0;i++) cnt1=i;
        for(int i=1;i<=cnt1;i++)
        {
            memset(tmp,false,sizeof tmp);
            for(int p=1;p<=s[i].siz;p++) sum[i]+=a[s[i].x[p]],tmp[s[i].x[p]]=true;
            for(int j=1;j<=m;j++)
                for(int p=1;p<=s[j].siz;p++) if(tmp[s[j].x[p]]) cnt[i][j]++;
        }
        for(int owo=1;owo<=q;owo++)
        {
            char opt; scanf("%c",&opt);
            if(opt=='?')
            {
                int k=ori[read()];
                if(k<=cnt1) printf("%lld
    ",sum[k]);
                else
                {
                    lint res=0;
                    for(int p=1;p<=s[k].siz;p++) res+=a[s[k].x[p]];
                    for(int i=1;i<=cnt1;i++) res+=add[i]*cnt[i][k];
                    printf("%lld
    ",res);
                }
            }
            if(opt=='+')
            {
                int k=ori[read()]; lint v=read();
                if(k<=cnt1) add[k]+=v;
                else for(int p=1;p<=s[k].siz;p++) a[s[k].x[p]]+=v;
                for(int i=1;i<=cnt1;i++) sum[i]+=v*cnt[i][k];
            }
        }
        return 0;
    }
    

    P.S.

    可以用vector<int>存储集合元素,直接开数组内存太大。
    要开long long啊啊啊啊啊!

  • 相关阅读:
    无标题
    UVA 11987 并查集删点
    屯题 (bestcoder #62~#75)
    codeforces 293E Close Vertices 点分治+滑窗+treap
    hdu4670 Cube number on a tree 点分治
    hdu4812 D Tree 点分治
    poj2112 Boatherds 点分治
    HDU 4866 Shooting 二分+主席树
    poj1741 Tree 点分治
    关于点分治。。。
  • 原文地址:https://www.cnblogs.com/VisJiao/p/8492216.html
Copyright © 2011-2022 走看看