COGS.1822.[AHOI2013]作业(莫队 树状数组/分块)

题目链接: COGSBZOJ3236
Upd: 树状数组实现的是单点加 区间求和,采用值域分块可以\(O(1)\)修改\(O(sqrt(n))\)查询。同BZOJ3809.

莫队为\(O(n^{1.5})\)次修改和\(O(n)\)次查询。
注意这两个需求并不平衡,所以在搭配数据结构时常使用分块而不是线段树。
(转自莫队复杂度分析 by Meiku Kazami)

1.莫队+树状数组

/*
每个[l,r]的询问中又多了[a,b]值的限制。原先now是所有种类的个数,所以用 莫队+树状数组做 
两问,维护两个树状数组 
O(m*sqrt(n)*logn)
*/
#include<cmath>
#include<cstdio>
#include<cctype>
#include<algorithm>
using namespace std;
const int N=1e5+5,M=1e6+5;

int n,m,size,A[N],t1[N],t2[N],times[N],Ans1[M],Ans2[M];
struct Ques
{
    int l,r,a,b,id;
    bool operator <(const Ques &x)const
    {
        return l/size==x.l/size ? r<x.r : l/size<x.l/size;
    }
}q[M];

inline int read()
{
    int now=0,f=1;register char c=getchar();
    for(;!isdigit(c);c=getchar())
      if(c=='-') f=-1;
    for(;isdigit(c);now=now*10+c-'0',c=getchar());
    return now*f;
}

inline int lb(int x)
{
    return x&-x;
}
void Update(int p,int v,int *t)
{
    while(p<=n)
        t[p]+=v, p+=lb(p);
}
int Query(int p,int *t)
{
    int res=0;
    while(p)
        res+=t[p], p-=lb(p);
    return res;
}
void Add(int p)
{
    Update(A[p],1,t1);
    if(!times[A[p]]) Update(A[p],1,t2);
    ++times[A[p]];
}
void Subd(int p)
{
    Update(A[p],-1,t1);
    --times[A[p]];
    if(!times[A[p]]) Update(A[p],-1,t2);
}

int main()
{
    freopen("ahoi2013_homework.in","r",stdin);
    freopen("ahoi2013_homework.out","w",stdout);

    n=read(),m=read();
    size=sqrt(n);
    for(int i=1;i<=n;++i)
        A[i]=read();
    for(int i=1;i<=m;++i)
        q[i].l=read(), q[i].r=read(), q[i].a=read(), q[i].b=read(), q[i].id=i;
    sort(q+1,q+1+m);
    for(int l=1,r=0,i=1;i<=m;++i)
    {
        int ln=q[i].l,rn=q[i].r;
        while(l<ln) Subd(l++);
        while(l>ln) Add(--l);
        while(r<rn) Add(++r);
        while(r>rn) Subd(r--);
        Ans1[q[i].id]=Query(q[i].b,t1)-Query(q[i].a-1,t1),//注意值的区间是[a,b]不是[l,r] 
        Ans2[q[i].id]=Query(q[i].b,t2)-Query(q[i].a-1,t2);
//      printf("%d:%d %d ans1:%d ans2:%d\n",q[i].id,ln,rn,Ans1[q[i].id],Ans2[q[i].id]);
    }
    for(int i=1;i<=m;++i)
        printf("%d %d\n",Ans1[i],Ans2[i]);

    fclose(stdin);fclose(stdout);
    return 0;
}

2.莫队+值域分块

#include <cmath>
#include <cstdio>
#include <cctype>
#include <algorithm>
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
const int N=1e5+5,MAXIN=2e6;

int n,m,size,Ans1[N*10],Ans2[N*10],A[N],tm[N],bel[N],sum1[500],sum2[500];
char IN[MAXIN],*SS=IN,*TT=IN;
struct Ask
{
    int l,r,a,b,id;
    bool operator <(const Ask &a)const
    {
        return l/size==a.l/size ? r<a.r : l/size<a.l/size;//更快 WTF 
//      return l/size==a.l/size?((l-1)/size&1 ? r>a.r : r<a.r):l/size<a.l/size;
    }
}q[N*10];

inline int read()
{
    int now=0;register char c=gc();
    for(;!isdigit(c);c=gc());
    for(;isdigit(c);now=now*10+c-'0',c=gc());
    return now;
}
int Query(int *s,int l,int r,bool f)
{
    int res=0,tmp=std::min(r,bel[l]*size);
    for(int i=l; i<=tmp; ++i) res+= f?(tm[i]>0):tm[i];
    if(bel[l]!=bel[r])
        for(int i=(bel[r]-1)*size+1; i<=r; ++i)
            res+= f?(tm[i]>0):tm[i];
    for(int i=bel[l]+1; i<bel[r]; ++i) res+=s[i];
    return res;
}
void Add(int p)
{
    if(++tm[p]==1) ++sum2[bel[p]];
    ++sum1[bel[p]];
}
void Subd(int p)
{
    if(!--tm[p]) --sum2[bel[p]];
    --sum1[bel[p]];
}

int main()
{
    n=read(), m=read(), size=sqrt(n);;//size=n/sqrt(m*2/3) //也没有更快 数组大小还要注意 
    for(int i=1; i<=n; ++i) bel[i]=(i-1)/size+1, A[i]=read();
    for(int i=1; i<=m; ++i) q[i].l=read(),q[i].r=read(),q[i].a=read(),q[i].b=read(),q[i].id=i;
    std::sort(q+1,q+1+m);
    for(int l=1,r=0,ln,rn,i=1; i<=m; ++i)
    {
        int ln=q[i].l,rn=q[i].r;
        while(l<ln) Subd(A[l++]);
        while(l>ln) Add(A[--l]);
        while(r<rn) Add(A[++r]);
        while(r>rn) Subd(A[r--]);
        Ans1[q[i].id]=Query(sum1,q[i].a,q[i].b,0), Ans2[q[i].id]=Query(sum2,q[i].a,q[i].b,1);
    }
    for(int i=1; i<=m; ++i) printf("%d %d\n",Ans1[i],Ans2[i]);

    return 0;
}

转载于:https://www.cnblogs.com/SovietPower/p/8435094.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值