码迷,mamicode.com
首页 > 其他好文 > 详细

BZOJ.2724.[Violet 6]蒲公英(静态分块)

时间:2018-02-27 23:22:05      阅读:144      评论:0      收藏:0      [点我收藏+]

标签:cstring   def   efi   crete   ret   swa   ref   body   inline   

题目链接

区间众数 强制在线
考虑什么样的数会成为众数
如果一个区间S1的众数为x,那么S1与新区间S2的并的众数只会是x或S2中的数
所以我们可以分块先预处理f[i][j]表示第i到第j块的众数
对于零散部分,我们还需要知道它们在区间中的出现次数。这部分至多有2sqrt(n)个
由于没有修改,离散化后对于每个数x开一个vector,把x出现位置push_back进去,查x时二分即可。
像普通分块一样更新即可。神tm写了一堆特判
这个块怎么算。。没算出来。

//2520kb    19652ms 这么慢。。
#include <cmath>
#include <cctype>
#include <cstdio>
#include <vector>
#include <cstring>
#include <algorithm>
#define gc() getchar()
const int N=4e4+5,S=205;

int n,m,size,f[S][S],t[S][S],A[N],ref[N],bel[N],tm[N];
std::vector<int> v[N];

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 Find(int x,int r)
{
    int l=1,mid;
    while(l<r)
        if(ref[mid=l+r>>1]>=x) r=mid;
        else l=mid+1;
    return l;
}
void Discrete()
{
    for(int i=1; i<=n; ++i) ref[i]=A[i]=read();
    std::sort(ref+1,ref+1+n);
    int cnt=1;
    for(int i=2; i<=n; ++i)
        if(ref[i]!=ref[i-1]) ref[++cnt]=ref[i];
    for(int i=1; i<=n; ++i) A[i]=Find(A[i],cnt);
}
void Init()
{
    Discrete();
    for(int i=1; i<=n; ++i) bel[i]=(i-1)/size+1;
    for(int i=1; i<=n; ++i) v[A[i]].push_back(i);
    for(int i=1; i<=bel[n]; ++i)
    {
        memset(tm,0,sizeof tm);
        int id=0,mxt=0;
        for(int j=i; j<=bel[n]; ++j)
        {
            int r=std::min(size*j,n);
            for(int k=(j-1)*size+1; k<=r; ++k)
                if(++tm[A[k]]>mxt) id=A[k],mxt=tm[id];
                else if(tm[A[k]]==mxt && A[k]<id) id=A[k];
            f[i][j]=id, t[i][j]=mxt;
        }
    }
}
int Lower(int id,int x)
{
    int l=0,r=v[id].size()-1,mid;
    while(l<r)
        if(v[id][mid=l+r>>1]>=x) r=mid;
        else l=mid+1;
    return l;
}
int Upper(int id,int x)
{
    int l=0,r=v[id].size(),mid;
    while(l+1<r)
    {
        if(v[id][mid=l+r>>1]<=x) l=mid;
        else r=mid;
    }
    return l;
}
inline int Query(int l,int r,int id){
    if(!id) return 0;
    return Upper(id,r)-Lower(id,l)+1;
}
void Update(int l,int r,int L,int R,int &id,int &mxt)
{
    for(int tmp,i=l; i<=r; ++i)
        if((tmp=Query(L,R,A[i]))>mxt||(tmp==mxt&&id>A[i]))
            id=A[i], mxt=tmp;
}
int Solve(int l,int r)
{
    int id=f[bel[l]+1][bel[r]-1],mxt=t[bel[l]+1][bel[r]-1];//mxt=Query(l,r,id);//这样id可能不存在 为0,Query中必须要特判 
    Update(l,std::min(r,bel[l]*size),l,r,id,mxt);
    if(bel[l]!=bel[r]) Update((bel[r]-1)*size+1,r,l,r,id,mxt);
    return id;
}

int main()
{
    n=read(),m=read(); size=sqrt(n);//200
    Init();
    int res=0,l,r;
    while(m--)
    {
        l=(read()+res-1)%n+1, r=(read()+res-1)%n+1;
        if(l>r) std::swap(l,r);
        printf("%d\n",res=ref[Solve(l,r)]);
    }
    return 0;
}

BZOJ.2724.[Violet 6]蒲公英(静态分块)

标签:cstring   def   efi   crete   ret   swa   ref   body   inline   

原文地址:https://www.cnblogs.com/SovietPower/p/8480647.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!