zoukankan      html  css  js  c++  java
  • HDU

    HDU - 3333

    题目大意:求区间不同类型的数字之和。

    离线:将询问按 r 值排序,拿到一个询问的时候,我们将比位置在r之前的数字全部加入树状数组,加入数字的规则如下:

    假设加入的数字为x,如果x之前没有出现过,直接加入,否则删除前一个x,再加入x,这样保证了树状数组里面一类数只有一个,并且都是最靠右的数,

    这样显然是最优并且正确的。

    #include<bits/stdc++.h>
    #define fi first
    #define se second
    #define mk make_pair
    #define pii pair<int,int>
    #define read(x) scanf("%d",&x)
    #define sread(x) scanf("%s",x)
    #define dread(x) scanf("%lf",&x)
    #define lread(x) scanf("%lld",&x)
    using namespace std;
    
    typedef long long ll;
    const int inf=0x3f3f3f3f;
    const int INF=0x3f3f3f3f3f3f3f3f;
    const int N=3e5+7;
    const int M=1e5+7;
    
    int n,a[N],pre[N];
    ll ans[N];
    struct Qus
    {
        int l,r,id;
        bool operator < (const Qus &rhs)const
        {
            if(r==rhs.r)
                return l<rhs.l;
            return r<rhs.r;
        }
    }qus[M];
    struct BIT
    {
        ll a[N];
        void init(){memset(a,0,sizeof(a));}
        void modify(int pos,int v)
        {
            for(int i=pos;i<=n;i+=i&-i)
                a[i]+=v;
        }
        ll sum(int pos)
        {
            ll ans=0;
            for(int i=pos;i;i-=i&-i)
                ans+=a[i];
            return ans;
        }
    }bit;
    int main()
    {
        int T; read(T);
        while(T--)
        {
            map<int,int> mp;
            bit.init();
            read(n);
            for(int i=1;i<=n;i++)
            {
                read(a[i]);
                pre[i]=mp[a[i]];
                mp[a[i]]=i;
            }
            int q; read(q);
            for(int i=1;i<=q;i++)
                read(qus[i].l),read(qus[i].r),qus[i].id=i;
            sort(qus+1,qus+q+1);
            int now=1;
            for(int i=1;i<=q;i++)
            {
                while(now<=n && now<=qus[i].r)
                {
                    if(pre[now])
                        bit.modify(pre[now],-a[now]);
                    bit.modify(now,a[now]);
                    now++;
                }
                ans[qus[i].id]=bit.sum(qus[i].r)-bit.sum(qus[i].l-1);
            }
            for(int i=1;i<=q;i++)
                printf("%lld
    ",ans[i]);
        }
        return 0;
    }
    /*
    */
    View Code

    在线:主席树,root [r] 表示,第加入第r个数之后的版本的根,root [r]为根的树保存的是每类数最靠右的那一个,思路和离线差不多。

    #include<bits/stdc++.h>
    #define fi first
    #define se second
    #define mk make_pair
    #define pii pair<int,int>
    #define read(x) scanf("%d",&x)
    #define sread(x) scanf("%s",x)
    #define dread(x) scanf("%lf",&x)
    #define lread(x) scanf("%lld",&x)
    using namespace std;
    
    typedef long long ll;
    const int inf=0x3f3f3f3f;
    const int INF=0x3f3f3f3f3f3f3f3f;
    const int N=3e5+7;
    const int M=51;
    
    int n,pre[N],root[N];
    map<int,int> mp;
    struct Chairman_tree
    {
        int cnt;
        struct node{
            int l,r;
            ll sum;
        }a[20*N];
        void update(int l,int r,int &x,int y,int pos,int v)
        {
            a[++cnt]=a[y];
            x=cnt; a[x].sum+=v;
            if(l==r) return;
            int mid=(l+r)>>1;
            if(pos<=mid)
                update(l,mid,a[x].l,a[y].l,pos,v);
            else
                update(mid+1,r,a[x].r,a[y].r,pos,v);
        }
        ll query(int l,int r,int pos,int x)
        {
            if(l>=pos)
                return a[x].sum;
            if(r<pos)
                return 0;
            int mid=(l+r)>>1;
            return query(l,mid,pos,a[x].l)+query(mid+1,r,pos,a[x].r);
        }
    }seg;
    int main()
    {
        int T; read(T);
        while(T--)
        {
            map<int,int> mp;
            seg.cnt=0;
            read(n);
            for(int i=1;i<=n;i++)
            {
                int x; read(x);
                if(mp[x])
                {
                    seg.update(1,n,root[n+1],root[i-1],mp[x],-x);
                    seg.update(1,n,root[i],root[n+1],i,x);
                }
                else
                    seg.update(1,n,root[i],root[i-1],i,x);
                mp[x]=i;
            }
            int q; read(q);
            while(q--)
            {
                int l,r;
                read(l); read(r);
                printf("%lld
    ",seg.query(1,n,l,root[r]));
            }
        }
        return 0;
    }
    View Code

    莫队

    #include<bits/stdc++.h>
    #define fi first
    #define se second
    #define mk make_pair
    #define pii pair<int,int>
    #define read(x) scanf("%d",&x)
    #define sread(x) scanf("%s",x)
    #define dread(x) scanf("%lf",&x)
    #define lread(x) scanf("%lld",&x)
    using namespace std;
    
    typedef long long ll;
    const int inf=0x3f3f3f3f;
    const int INF=0x3f3f3f3f3f3f3f3f;
    const int N=3e5+7;
    const int M=1e5+7;
    
    int n,q,B,a[N],cnt[N],num[N];
    ll ret,ans[N];
    struct Qus
    {
        int l,r,id;
        bool operator < (const Qus &rhs)const
        {
            if(l/B==rhs.l/B)
                return r/B<rhs.r/B;
            return l/B<rhs.l/B;
        }
    }qus[N];
    void update(int pos,int op)
    {
        if(op==1)
        {
            if(cnt[num[pos]]==0)
                ret+=a[pos];
            cnt[num[pos]]++;
        }
        else
        {
            if(cnt[num[pos]]==1)
                ret-=a[pos];
            cnt[num[pos]]--;
        }
    }
    void solve()
    {
        B=sqrt(n);
        memset(cnt,0,sizeof(cnt));
        sort(qus+1,qus+1+q);
        int l=1,r=0; ret=0;
        for(int i=1;i<=q;i++)
        {
            int L=qus[i].l,R=qus[i].r;
            while(r<R) update(++r,1);
            while(l>L) update(--l,1);
            while(r>R) update(r--,-1);
            while(l<L) update(l++,-1);
            ans[qus[i].id]=ret;
        }
    }
    int main()
    {
        int T; read(T);
        while(T--)
        {
            map<int,int> mp;
            int tot=0;
            read(n);
            for(int i=1;i<=n;i++)
            {
                read(a[i]);
                if(!mp[a[i]])
                    mp[a[i]]=num[i]=++tot;
                else
                    num[i]=mp[a[i]];
            }
            read(q);
            for(int i=1;i<=q;i++)
                read(qus[i].l),read(qus[i].r),qus[i].id=i;
            solve();
            for(int i=1;i<=q;i++)
                printf("%lld
    ",ans[i]);
        }
        return 0;
    }
    /*
    */
    View Code
  • 相关阅读:
    正定矩阵与半正定矩阵定义与判别
    LQR (线性二次型调节器)的直观推导及简单应用
    simulink模块使用方式
    无人驾驶领域现有的主流智能车辆仿真软件具体情况介绍
    什么是在环测试
    C# Dictionary 是否包含key
    jquery.显示隐藏切换
    easyui-dialog
    Visual Studio 命令行wsdl生成C#操作类
    使用C#创建Windows服务
  • 原文地址:https://www.cnblogs.com/CJLHY/p/8504714.html
Copyright © 2011-2022 走看看