[日常摸鱼]bzoj2724蒲公英-分块

区间众数经典题~

http://begin.lydsy.com/JudgeOnline/problem.php?id=4839
这里可以提交~

题意大概就是没有修改的询问区间众数,如果有一样的输出最小的,强制在线,$n \leq 4*10^4,a_i \leq 10^9$。



log数据结构脑补一遍好像没什么可以做的,数据范围我们可以分块!

不过分块之前肯定要离散化一下,而且还要保存离散化前的数据(因为要回答的是出现最多的数),离散化的方法在上一篇博客里面~

假设分成$L$块,每块大小$s=\lfloor n/L \rfloor$,预处理出每一块的起点和终点$st[],ed[]$,然后再预处理出$O(L^2)$个连续的块里的答案(每种颜色的出现次数,最大次数和对应的颜色),这样预处理的复杂度是$O(n*L^2)$。

然后对于一个区间的询问$[x,y]$,一定能拆成连续的若干个块(假设对应的块是$[p,q]$)和至多两个不完整的块,对于中间连续的块直接用我们之前预处理的答案。两边的暴力更新。

一种更新方法是把两边的颜色都加到中间完整的一段去,然后更新答案,更新完之后再减回去。复杂度$O(\frac{mn}{L})$,于是总复杂度$O(n*L^2+\frac{mn}{L})$,$m,n$同阶,均值不等式告诉我们当$L=n^{\frac{1}{3}}$的时候复杂度取最小值为$O(n^{\frac{5}{3}})$

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#define rep(i,n) for(register int i=1;i<=n;i++)
#define REP(i,a,b) for(register int i=a;i<=b;i++)
#define debug(x) printf("%s =  %d , ",#x,x)
using namespace std;
const int T=40;
const int N=40005;
inline int read()
{
    int s=0,f=1;char c=getchar();
    while(c<‘0‘||c>‘9‘){if(c==‘-‘)f=0;c=getchar();}
    while(c>=‘0‘&&c<=‘9‘){s=s*10+c-‘0‘;c=getchar();}
    return f?s:-s;
}
int n,Q,L,s,ans,cnt,tot,num,p,q;
int w[N],tw[N],v[N],b[N],c[T][T][N],f[T][T],d[T][T],st[T],ed[T];
inline void prework()
{
    n=read();Q=read();
    rep(i,n)w[i]=read();

    L=(int)pow(n,1.0/3.0);
    if(L)s=n/L;
    rep(i,L)st[i]=(i-1)*s+1,ed[i]=i*s;
    if(ed[L]!=n)st[L+1]=ed[L]+1,ed[++L]=n;

    memcpy(tw,w,sizeof tw);sort(tw+1,tw+n+1);
    rep(i,n)if(i==1||tw[i]!=tw[i-1])v[++tot]=tw[i];
    rep(i,n)b[i]=lower_bound(v+1,v+tot+1,w[i])-v;
    REP(i,1,L)REP(j,i,L)
    {
        rep(k,tot)c[i][j][k]=c[i][j-1][k];
        REP(k,st[j],ed[j])c[i][j][b[k]]++;
        rep(k,tot)if(c[i][j][k]>f[i][j]||(c[i][j][k]==f[i][j]&&tw[k]<d[i][j]))
            f[i][j]=c[i][j][k],d[i][j]=k;
    }
}
inline void updata(int i)
{
    c[p][q][b[i]]++;
    if(c[p][q][b[i]]>cnt||(c[p][q][b[i]]==cnt&&b[i]<num))cnt=c[p][q][b[i]],num=b[i];
}
inline int solve(int x,int y)
{
    if(x>y)swap(x,y);
    int l,r;
    for(register int i=1;i<=L;i++)if(x<=ed[i]){l=i;break;}
    for(register int i=L;i>=1;i--)if(y>=st[i]){r=i;break;}
    if(l+1<=r-1)p=l+1,q=r-1;
    else p=q=0;
    cnt=f[p][q];num=d[p][q];
    if(l==r)
    {
        REP(i,x,y)updata(i);
        REP(i,x,y)c[p][q][b[i]]--;
    }else
    {
        REP(i,x,ed[p-1])updata(i);
        REP(i,st[q+1],y)updata(i);
        REP(i,x,ed[p-1])c[p][q][b[i]]--;
        REP(i,st[q+1],y)c[p][q][b[i]]--;
    }
    return v[num];
}

int main()
{
    prework();
    rep(i,Q)
    {
        int l,r;
        l=read();r=read();
        ans=solve((l+ans-1)%n+1,(r+ans-1)%n+1);
        printf("%d\n",ans);
    }
    return 0;
}

原文地址:https://www.cnblogs.com/yoooshinow/p/8413132.html

时间: 2024-08-29 21:18:33

[日常摸鱼]bzoj2724蒲公英-分块的相关文章

[日常摸鱼]Uva11178Morley&#39;s Theorem-几何

题意:给一个$\Delta ABC$,分别做三个角的三等分线相交成$\Delta DEF$,求出$D,E,F$的坐标. 直接根据题意模拟 #include<cstdio> #include<cmath> #include<algorithm> using namespace std; struct Point { double x,y; Point(double x1=0,double y1=0){x=x1;y=y1;} }; typedef Point Vector;

[日常摸鱼]JSOI2008最大数

校运会的时候随手抽的题- 一句话题意 维护一个序列,初始为空,要求滋兹: 1.查询这个序列末尾$x$个数的最大值 2.设上一次查询的答案为$t$(如果还没查询$t=0$),在末尾插入一个数$(x+t)mod d$,$d$为给定常数 很容易想到用线段树做:记录序列的末尾,然后直接单点修改区间查询 本来想随便写完就过了的-然后一直爆零- 因为我写了一句 while(n--) 然后这题应该就没什么要注意的地方了233 贴代码 #include<cstdio> #include<cstring&

[日常摸鱼]UVA393 The Doors 简单计算几何+最短路

The  Boy Next   Doors 题意:给定一个固定大小的房间($x,y$的范围都是$[0,10]$),有$n$个墙壁作为障碍(都与横坐标轴垂直),每个墙壁都有两扇门分别用四个点来描述,起点终点固定在$(0,5)$和$(10,5)$,求起点到终点的最短路长度,$n<=18$ 题解: 我们把每堵墙的每一"段"作为一条线段,对任意两点$u,v$,如果两点间的连线不和其他线段相交,那我们从$u$走到$v$的最短距离就是他们的欧几里得距离,对所有点对都这么做一遍,处理出所有能够

[日常摸鱼]杜教筛

因为博主比较菜所以可能一些地方写的有问题或者不清楚,以及我的废话有点多- 在这里先感谢下小伙伴ww @MoebiusMeow 的帮助~ 参考资料: [1]浅谈一类积性函数的前缀和(skywalkert) [2]杜教筛--省选前的学习1(_rqy) (下面约定$[p]$表示满足条件$p$时为1不满足为0,以及下面说的函数都是数论函数) 前置技能(一些定义) 数论函数:若$f:Z^{+} \rightarrow C$,则称$f$为数论函数 积性函数:若一个数论函数$f(n)$对于所有$m_1 \bo

[日常摸鱼]字符串相关

最近开始学字符串-之前暑假听JOHNKROM大爷讲这个的时候就很懵(当时根本没碰过这个东西-) 这里大概会有一些相关的算法和自己做的题,还有存一些模板啥的. 由于我太弱了所以可能不会写有关原理的东西(或者等我哪天会了有空再补)如果是刚学的话还是去网上找教程吧-我就存个题还有模板啥的orz Trie字典树(Trie),根节点为空节点,其他节点(或边上)放字母,然后节点上还能储存一些其他信息. 1.一个类似模板题的东西:luogu2580 建一颗Trie,把所有名字插进去,结束的地方打个标记,具体见

[日常摸鱼]luogu1613跑路

新年A的第一道题2333 https://www.luogu.org/problemnew/show/P1613 题意:给一张有向图,每条边长为1,每个单位时间只能走$2^k$的长度,$k$可以任意选,问从起点到终点的最短时间 我好菜呀一看到题还是不会做orz 我们用一个$G[i][j][k]$数组来表示$i$到$j$是否存在$2^k$的路径,用$dis[i][j]$表示$i$到$j$需要的最短时间,有连边的两个点就直接让$G[i][j][0]=1,dis[i][j]=1$,然后类似倍增那样子去

[日常摸鱼]luogu3398仓鼠找sugar-树链剖分

https://www.luogu.org/problemnew/show/P3398 题意:一颗$n$个点的树,$q$次询问两条链$(a,b),(c,d)$是否有交 树剖裸题orz 一开始的想法是求出$lca_1=lca(a,b),lca_2=lca(c,d)$,对于深度大的那个$lca$用dfs序判断是否在另一条链上,然后到现在都不知道为什么一直wa- 改成判断深度大的那个$lca$是否和另外两个点的其中某个点有父子关系好像就行了- #include<cstdio> #include<

[日常摸鱼][POI2000]病毒-Tire图(AC自动机)+dfs

https://www.luogu.org/problemnew/show/P2444 (没有bzoj权限号T_T) 字符串题对我这种傻逼来说真是太难了x 题意:输入$n$个01组成的模式串串,判断是否存在一个无限长的01串满足任何给定的模式串都不是这个串的子串,$n \leq 2000,\sum len(str) \leq 30000$ 首先用模式串构造出AC自动机(其实应该叫Trie图),题目要求的串如果存在,那就相当于在Trie图上存在一个环,这个环不包含任何一个模式串,所以只要构造出来T

[日常摸鱼]bzoj1502[NOI2005]月下柠檬树-简单几何+Simpson法

关于自适应Simpson法的介绍可以去看我的另一篇blog http://www.lydsy.com/JudgeOnline/problem.php?id=1502 题意:空间里圆心在同一直线上且底面与地面平行的若干个圆台和顶层的圆锥以$\alpha$的角度投影到地面,求投影的面积. (其实我是看po姐博客来的x) 首先把圆锥的顶点也看成一个半径为0的圆满,对于每个高度为$h$的圆投影下去的坐标是$h/tan(\alpha)$,半径不变,而对于圆台的侧面投影下去是上下底两个圆的切线. 关于两个圆