博客园 首页 私信博主 显示目录 隐藏目录 管理 动画

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

BZOJ
洛谷
DBZOJ


\(Description\)
给定长为\(n\)的序列,\(m\)次求区间众数,强制在线。
\(n\leq40000,m\leq50000\)

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

像普通分块一样更新即可。之前写了一堆特判 醉了

这个块大小怎么算。。没算出来。


//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;
}
posted @ 2018-02-27 20:03  SovietPower  阅读(182)  评论(0编辑  收藏  举报