Luogu P4168 [Violet]蒲公英 分块
这道题算是好好写了。写了三种方法。
有一个好像是$qwq$$N\sqrt(N)$的方法,,但是恳请大佬们帮我看看为什么这么慢$qwq$(后面的第三种)
注:$pos[i]$表示$i$属于第$pos[i]$块。
第一种是统计所有可能的块组成的区间中(第i块到第j块),每个数出现的次数,记做$f[i][j][k]$,和所有可能的块组成的区间的答案,记做$h[i][j]$。
然后每次先把整块的答案作为初始答案,然后对于散块中的每个值$vl$,暴力修改对应的$f[i][j][vl]$,更新答案。
当块长取$N^\frac{2}{3}$,时间复杂度$O(N^\frac{5}{3})$级。
#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<cctype>
#include<cstdlib>
#include<vector>
#include<queue>
#include<map>
#include<set>
#define ull unsigned long long
#define ll long long
#define R register int
#define pause (for(R i=1;i<=10000000000;++i))
#define OUT freopen("out.out","w",stdout);
using namespace std;
namespace Fread {
static char B[<<],*S=B,*D=B;
#define getchar() (S==D&&(D=(S=B)+fread(B,1,1<<15,stdin),S==D)?EOF:*S++)
inline int g() {
R ret=,fix=; register char ch; while(!isdigit(ch=getchar())) fix=ch=='-'?-:fix;
do ret=ret*+(ch^); while(isdigit(ch=getchar())); return ret*fix;
} inline bool isempty(const char& ch) {return ch<=||ch>=;}
inline void gs(char* s) {register char ch; while(isempty(ch=getchar())); do *s++=ch; while(!isempty(ch=getchar()));}
}using Fread::g; using Fread::gs;
const int N=,M=; int n,m,tot,T,lst;
int f[M][M][N],h[M][M],vl[N],a[N],b[N],pos[N];
inline void PRE() { R mx=,ans=;
for(R i=;i<=n;++i) pos[i]=(i-)/T+;
for(R j=,L=pos[n];j<=L;++j,mx=,ans=) for(R t=j;t<=L;++t) {
memcpy(f[j][t],f[j][t-],sizeof(f[j][t-]));
for(R i=(t-)*T+,LL=min(t*T,n);i<=LL;++i) ++f[j][t][a[i]];
for(R i=tot;i;--i) if(f[j][t][i]>=mx) mx=f[j][t][i],ans=i;
h[j][t]=ans;
}
}
signed main() {
#ifdef JACK
freopen("NOIPAK++.in","r",stdin);
OUT;
#endif
n=g(),m=g(),T=n/pow(n,1.0/);
for(R i=;i<=n;++i) a[i]=g(); memcpy(b,a,sizeof(a));
sort(b+,b+n+); tot=unique(b+,b+n+)-b-;
for(R i=;i<=n;++i) a[i]=lower_bound(b+,b+tot+,a[i])-b;
memcpy(vl,b,sizeof(int)*(tot+)); PRE();
for(R i=,l,r;i<=m;++i) { R mx=,ans=;
l=(g()+lst-)%n+,r=(g()+lst-)%n+; l>r?swap(l,r):(void);
R p=pos[l]+,q=pos[r]-; ans=h[p][q],mx=f[p][q][ans];
if(pos[l]==pos[r]) {
for(R i=l;i<=r;++i) { ++f[p][q][a[i]];
if(f[p][q][a[i]]>mx||(f[p][q][a[i]]==mx&&a[i]<ans)) mx=f[p][q][a[i]],ans=a[i];
} for(R i=l;i<=r;++i) --f[p][q][a[i]];
} else { ans=h[p][q],mx=f[p][q][ans];
for(R i=l,L=pos[l]*T;i<=L;++i) { ++f[p][q][a[i]];
if(f[p][q][a[i]]>mx||(f[p][q][a[i]]==mx&&a[i]<ans)) mx=f[p][q][a[i]],ans=a[i];
} for(R i=(pos[r]-)*T+;i<=r;++i) { ++f[p][q][a[i]];
if(f[p][q][a[i]]>mx||(f[p][q][a[i]]==mx&&a[i]<ans)) mx=f[p][q][a[i]],ans=a[i];
} for(R i=l,L=pos[l]*T;i<=L;++i) --f[p][q][a[i]];
for(R i=(pos[r]-)*T+;i<=r;++i) --f[p][q][a[i]];
} printf("%d\n",lst=vl[ans]);
}
}
第二种是预处理出所有可能的块组成的区间中(第$i$块到第$j$块)的答案$f[i][j]$,并且拿一个$vector$存每个数$vl$出现的位置$s[vl][1...n]$。
答案初始化为整块的答案,然后对于散块中的每个数$vl$,在$s[vl]$中二分出$[l,r]$的最小和最大的位置的下标,相减就是$[l,r]$有多少个$vl$,然后更新答案。
当块长取$\sqrt{\frac{N}{logN}}$,时间复杂度$O(N\sqrt{NlogN})$。
#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<cctype>
#include<cstdlib>
#include<vector>
#include<queue>
#include<map>
#include<set>
#define ull unsigned long long
#define ll long long
#define R register int
#define pause (for(R i=1;i<=10000000000;++i))
#define OUT freopen("out.out","w",stdout);
using namespace std;
namespace Fread {
static char B[<<],*S=B,*D=B;
#define getchar() (S==D&&(D=(S=B)+fread(B,1,1<<15,stdin),S==D)?EOF:*S++)
inline int g() {
R ret=,fix=; register char ch; while(!isdigit(ch=getchar())) fix=ch=='-'?-:fix;
do ret=ret*+(ch^); while(isdigit(ch=getchar())); return ret*fix;
} inline bool isempty(const char& ch) {return ch<=||ch>=;}
inline void gs(char* s) {register char ch; while(isempty(ch=getchar())); do *s++=ch; while(!isempty(ch=getchar()));}
}using Fread::g; using Fread::gs;
map<int,int> mp;
const int N=; int n,m,T,tot,lst;
vector<int> s[N];
#define pb push_back
int f[][],cnt[N],p[N],pos[N],a[N],b[N],vl[N];
inline void PRE(int p) { R ans=,mx=; memset(cnt,,sizeof(cnt));
for(R t=p,lim=pos[n];t<=lim;++t) {
for(R i=(t-)*T+,lim=min(n,t*T);i<=lim;++i) {
if(++cnt[a[i]]>mx||(cnt[a[i]]==mx&&a[i]<ans)) mx=cnt[a[i]],ans=a[i];
} f[p][t]=ans;
}
}
inline int calc(int l,int r,int x) {return upper_bound(s[x].begin(),s[x].end(),r)-lower_bound(s[x].begin(),s[x].end(),l);}
inline int solve(int l,int r) { R mx=,ret=;
if(pos[l]==pos[r]) { memset(cnt,,sizeof(cnt));
for(R i=l;i<=r;++i) if(++cnt[a[i]]>mx||(cnt[a[i]]==mx&&a[i]<ret)) ret=a[i],mx=cnt[a[i]];
} else { ret=f[pos[l]+][pos[r]-],mx=calc(l,r,ret);
for(R i=l,lim=pos[l]*T;i<=lim;++i) { R t=calc(l,r,a[i]);
if(t>mx||(t==mx&&a[i]<ret)) ret=a[i],mx=t;
} for(R i=(pos[r]-)*T+;i<=r;++i) { R t=calc(l,r,a[i]);
if(t>mx||(t==mx&&a[i]<ret)) ret=a[i],mx=t;
}
}
return ret;
}
signed main() {
#ifdef JACK
freopen("NOIPAK++.in","r",stdin);
OUT;
#endif
n=g(),m=g();//,T=n/sqrt(n*log2(n));
T=qpow(n,1.0/);
for(R i=;i<=n;++i) a[i]=g();
memcpy(b,a,sizeof(a)); sort(b+,b+n+);
tot=unique(b+,b+n+)-b-; memcpy(vl,b,sizeof(int)*(tot+));
for(R i=;i<=n;++i) a[i]=lower_bound(b+,b+tot+,a[i])-b,s[a[i]].pb(i);
for(R i=;i<=n;++i) pos[i]=(i-)/T+;
for(R i=;i<=pos[n];++i) PRE(i);
for(R i=,l,r;i<=m;++i) {
l=(g()+lst-)%n+,r=(g()+lst-)%n+; l>r?swap(l,r):(void);
printf("%d\n",lst=vl[solve(l,r)]);
}
}
第三种按理说是复杂度最优秀的,但是跑的不是很快$qwq$。
同第二种,预处理出所有可能的块组成的区间中(第i块到第j块)的答案$f[i][j]$,并且拿一个$vector$存每个数$vl$出现的位置$s[vl][1...n]$。
然后预处理$a[i]$是整个数列中的第几个$a[i]$,出第$1$到第$i$块中最靠后的$vl$是第几个$vl$,记为$d[i][vl]$,预处理出第$n$块到第$i$块中最靠前的$vl$是第几个$vl$,记为$h[i][vl]$。
对于左边散块中的$vl$,查一下$d[pos[r]-1][vl]$,和右边散块中是否有更靠后的$vl$(可以事先用一个数组存起来),然后可以求出这个区间有多少个$vl$(每一个$vl$是第几个$vl$已经知道了),更新答案。
当块长取$\sqrt{n}$时,时间复杂度为O(N\sqrt{N})级(不知道推没推错)。
#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<cctype>
#include<cstdlib>
#include<vector>
#include<queue>
#include<map>
#include<set>
#define ull unsigned long long
#define ll long long
#define R register int
#define pause (for(R i=1;i<=10000000000;++i))
#define OUT freopen("out.out","w",stdout);
using namespace std;
namespace Fread {
static char B[<<],*S=B,*D=B;
#define getchar() (S==D&&(D=(S=B)+fread(B,1,1<<15,stdin),S==D)?EOF:*S++)
inline int g() {
R ret=,fix=; register char ch; while(!isdigit(ch=getchar())) fix=ch=='-'?-:fix;
do ret=ret*+(ch^); while(isdigit(ch=getchar())); return ret*fix;
} inline bool isempty(const char& ch) {return ch<=||ch>=;}
inline void gs(char* s) {register char ch; while(isempty(ch=getchar())); do *s++=ch; while(!isempty(ch=getchar()));}
}using Fread::g; using Fread::gs;
const int N=,M=; int n,m,T,tot,lst;
vector<int> s[N];
#define pb push_back
int f[M][M],cnt[N],P[N],pos[N],a[N],b[N],vl[N],h[M][N],d[M][N],c[M][M];
inline void PRE(int p) { R ans=,mx=; memset(cnt,,sizeof(cnt));
for(R t=p,lim=pos[n];t<=lim;++t) {
for(R i=(t-)*T+,lim=min(n,t*T);i<=lim;++i) {
if(++cnt[a[i]]>mx||(cnt[a[i]]==mx&&a[i]<ans)) mx=cnt[a[i]],ans=a[i];
} f[p][t]=ans,c[p][t]=mx;
}
}
inline int solve(int l,int r) { R mx=,ret=,p=pos[l]+,q=pos[r]-;
if(pos[l]==pos[r]) { memset(cnt,,sizeof(cnt));
for(R i=l;i<=r;++i) if(++cnt[a[i]]>mx||(cnt[a[i]]==mx&&a[i]<ret)) ret=a[i],mx=cnt[a[i]];
} else { ret=f[p][q],mx=c[p][q]; memset(cnt,0x3f,sizeof(cnt));
for(R i=l,L=pos[l]*T;i<=L;++i) if(cnt[a[i]]==0x3f3f3f3f) cnt[a[i]]=P[i];
for(R i=q*T+;i<=r;++i) {
R tmp=P[i]+-min(cnt[a[i]],h[p][a[i]]);
if(tmp>mx||(tmp==mx&&a[i]<ret)) ret=a[i],mx=tmp;
cnt[a[i]]=P[i];
}
for(R i=l,L=pos[l]*T;i<=L;++i) {
R tmp=max((cnt[a[i]]==0x3f3f3f3f?:cnt[a[i]]),d[q][a[i]])-P[i]+;
if(tmp>mx||(tmp==mx&&a[i]<ret)) ret=a[i],mx=tmp;
}
} return ret;
}
signed main() {
#ifdef JACK
freopen("NOIPAK++.in","r",stdin);
OUT;
#endif
n=g(),m=g(); T=pow(n,/2.3);//好像更小一点更快(也不是越小越快)
for(R i=;i<=n;++i) a[i]=g();
memcpy(b,a,sizeof(a)); sort(b+,b+n+);
tot=unique(b+,b+n+)-b-; memcpy(vl,b,sizeof(int)*(tot+));
for(R i=;i<=n;++i) a[i]=lower_bound(b+,b+tot+,a[i])-b,s[a[i]].pb(i);
for(R i=;i<=n;++i) pos[i]=(i-)/T+; for(R i=;i<=n;++i) P[i]=++cnt[a[i]];
memset(cnt,,sizeof(cnt)); memset(h[pos[n]+],0x3f,sizeof(h[pos[n]+]));
for(R t=pos[n];t;--t) { memcpy(h[t],h[t+],sizeof(h[t+]));
for(R i=min(n,t*T);i>(t-)*T;--i) h[t][a[i]]=P[i];
} for(R t=;t<=pos[n];++t) { memcpy(d[t],d[t-],sizeof(d[t-]));
for(R i=(t-)*T+,L=t*T;i<=L;++i) d[t][a[i]]=P[i];
} for(R i=;i<=pos[n];++i) PRE(i);
for(R i=,l,r;i<=m;++i) {
l=(g()+lst-)%n+,r=(g()+lst-)%n+; l>r?swap(l,r):(void);
printf("%d\n",lst=vl[solve(l,r)]);
}
}
2019.06.28
Luogu P4168 [Violet]蒲公英 分块的更多相关文章
- luogu P4168 [Violet]蒲公英
嘟嘟嘟 分块经典题竟然是一道黑题…… 分块求区间众数的大体思想是对于询问区间[L, R],预处理出这中间的整块的众数,然后统计两边零散的数在[L, R]中出现的次数,最后取出现次数最多且最小的数. 因 ...
- 洛谷 P4168 [Violet]蒲公英 解题报告
P4168 [Violet]蒲公英 题目背景 亲爱的哥哥: 你在那个城市里面过得好吗? 我在家里面最近很开心呢.昨天晚上奶奶给我讲了那个叫「绝望」的大坏蛋的故事的说!它把人们的房子和田地搞坏,还有好多 ...
- p4168 [Violet]蒲公英(分块)
区间众数的重题 和数列分块入门9双倍经验还是挺好的 然后开O2水过 好像有不带log的写法啊 之后在补就是咕咕咕 // luogu-judger-enable-o2 #include <cstd ...
- [Violet]蒲公英 分块
发现写算法专题老是写不动,,,, 所以就先把我在luogu上的题解搬过来吧! 题目大意:查询区间众数,无修改,强制在线 乍一看是一道恐怖的题,仔细一看发现并没有那么难: 大致思路是这样的,首先我们要充 ...
- P4168 [Violet]蒲公英 区间众数
$ \color{#0066ff}{ 题目描述 }$ 在乡下的小路旁种着许多蒲公英,而我们的问题正是与这些蒲公英有关. 为了简化起见,我们把所有的蒲公英看成一个长度为n的序列 \((a_1,a_2.. ...
- P4168 [Violet]蒲公英
神仙分块题?其实还是很简单的,res[i][j]表示第i块到第j块的众数,然后再用sum[i][j]表示前i块中j这个种类出现的次数,然后分块瞎搞就行了,感觉我写的十分简洁,好评( //author ...
- BZOJ2724 [Violet]蒲公英 分块
题目描述 经典区间众数题目 然而是权限题,所以题目链接放Luogu的 题解 因为太菜所以只会$O(n*\sqrt{n}+n*\sqrt{n}*log(n))$的做法 就是那种要用二分的,并不会clj那 ...
- [洛谷P4168][Violet]蒲公英
题目大意:有$n(n\leqslant4\times10^4)$个数,$m(m\leqslant5\times10^4)$个询问,每次问区间$[l,r]$内的众数,若相同输出最小的,强制在线. 题解: ...
- 洛谷 P4168 [Violet] 蒲公英
历尽千辛万苦终于AC了这道题目... 我们考虑1个区间\([l,r]\), 被其完整包含的块的区间为\([L,R]\) 那么众数的来源? 1.\([l,L)\)或\((R,r]\)中出现的数字 2.\ ...
随机推荐
- 使用pycharm开发web——django2.1.5(四)视图和模板相关
刘老师说这块很重要..... 应该是很重要,大概看了一下,这里面关于views中函数作用,大概看来可能就是相应请求,传入数据和跳转,基本功能上貌似这些框架都差不多吧(其实我并没用过3个框架以上.... ...
- FastJson 介绍
Json详解 Json是一种轻量级的数据交换格式,采用一种“键:值”对的文本格式来存储和表示数据,在系统交换数据过程中常常被使用,是一种理想的数据交换语言.在使用 Java 做 Web 开发时,不可避 ...
- 你除了在客户端上会使用Cookie,还能使用哪些可以作为数据缓存呢?
问题如标题,直奔主题.介绍下另两种缓存. 1.sessionStorage.localStorage localStorage: 是一种你不主动清除它,它会一直将存储数据存储在客户端的存储方式,即使你 ...
- python以不同方式打印输出九九乘法表
参考:http://www.cnblogs.com/suiy-160428/p/5594389.htmlpython输出 9*9 乘法口诀表 矩形输出九九乘法表: for i in range(1,1 ...
- hdu 2610 2611 dfs的判重技巧
对于全排列枚举的数列的判重技巧 1:如果查找的是第一个元素 那么 从0开始到当前的位置看有没有出现过这个元素 出现过就pass 2: 如果查找的不是第一个元素 那么 从查找的子序列当前位置的前一个元素 ...
- 用于Linq的去重 Distinct
/// <summary> /// 用于Linq的去重,扩展方法需要放到静态类中 /// </summary> ...
- 设计模式 -- MVC
MVC 在Web中应用是常见的了,成为基础应用模式. 不好的用法是把业务写在C 中,M只是失血模型. 应该要重M 轻C,业务写在M中,但是这样有问题了.View 会引用Model,那么View会看到M ...
- JS基础_函数的参数
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- FFmpeg里面的时间单位
pts单位:1/90 ms(每个单位代表1/90 ms) RTP包头有个STAMP 对于视频 STAMP/90 就是 PTS (毫秒) 对于音频 STAMP/samplerate * 1000 才是 ...
- splice与slice区别
共同点:均是删除数组元素并返回结果. 区别:splice会改变原数组,而slice不会.并且splice会导致数组塌陷. 数组塌陷:使用splice删除元素时,剩余的数组元素索引的顺讯会改变. let ...