JZOJ5428. 【NOIP2017提高A组集训10.27】查询
来源:互联网 发布:淘宝旗舰店模板价格表 编辑:程序博客网 时间:2024/05/21 09:57
Description
给出一个长度为n的序列a[]
给出q组询问,每组询问形如(x,y),求a序列的所有区间中,数字x的出现次数与数字y的出现次数相同的区间有多少个
Input
第一行两个数n和q
第二行n个数a[i]
接下来q行,每行两个数x,y表示一组询问
Output
q行,每行一个数表示对应询问的答案
Sample Input
3 2
1 2 1
1 2
4 5
Sample Output
2
6
Data Constraint
对于30%的数据,1<=n<=100,1<=q<=1000
对于另外30%的数据,序列中只有最多50种不同的颜色且1<=n<=1000
对于100%的数据,1<=n<=8000,1<=q<=500000,1<=x,y,a[i]<=10^9
题解
因为要求区间,自然想到前缀和。
从头开始枚举,遇到x就+1,遇到y就-1,
那么前缀和相同的都棵两两构成一个区间。
如果对于每个询问都O(n)枚举前缀和,
那样是过不了的。
可以发现,这个前缀和只与x和y的位置有关。
对于每一个数,都可以维护一个数,表示下一个与它相同的数在哪里。
code
#include<queue>#include<cstdio>#include<iostream>#include<algorithm>#include <cstring>#include <string.h>#include <cmath>#include <math.h>#define ll long long#define N 8003#define db double#define P putchar#define G getchar#define mo 23333using namespace std;char ch;void read(int &n){ n=0; ch=G(); while((ch<'0' || ch>'9') && ch!='-')ch=G(); ll w=1; if(ch=='-')w=-1,ch=G(); while('0'<=ch && ch<='9')n=(n<<3)+(n<<1)+ch-'0',ch=G(); n*=w;}int max(int a,int b){return a>b?a:b;}ll min(ll a,ll b){return a<b?a:b;}ll abs(ll x){return x<0?-x:x;}ll sqr(ll x){return x*x;}void write(ll x){if(x>9) write(x/10);P(x%10+'0');}void writeln(ll x){write(x);P('\n');}int gcd(int x,int y){return y==0?x:gcd(y,x%y);}int a[N],nxt[N],b[mo+10],s,n,t[N*2],ans;int l,r,f[N][N],x,y,m,h[mo],t1,t2,tt;void ins(int x){ int y=x%mo; while(h[y]!=0 && h[y]!=x)y=(y+1)%mo; h[y]=x;}int find(int x){ int y=x%mo; while(h[y]!=0 && h[y]!=x)y=(y+1)%mo; return h[y]==x?y:mo+1;}int main() { freopen("query.in","r",stdin); freopen("query.out","w",stdout); read(n);read(m); for(int i=1;i<=n;i++) read(a[i]),a[i]++,ins(a[i]); for(int i=n;i;i--) { x=find(a[i]); nxt[i]=b[x]; b[x]=i; } b[mo+1]=n+1; memset(f,128,sizeof(f)); for(int i=1;i<=m;i++) { read(x);read(y);x++;y++; if(f[b[find(x)]][b[find(y)]]<0) { s=l=r=n;ans=tt=0;memset(t,0,sizeof(t)); t1=b[find(x)];t2=b[find(y)]; while(t1>0 || t2>0) { if((t1<t2 && t1!=0) || t2==0) { t[s]+=t1-tt; s++;r=max(r,s); tt=t1;t1=nxt[t1]; } else { t[s]+=t2-tt; s--;l=min(l,s); tt=t2;t2=nxt[t2]; } } t[s]+=n-tt+1; for(int j=l;j<=r;j++) ans+=t[j]*(t[j]-1)/2; f[b[find(x)]][b[find(y)]]=ans; } writeln(f[b[find(x)]][b[find(y)]]); }}
阅读全文
0 0
- 【JZOJ5428】【NOIP2017提高A组集训10.27】查询
- JZOJ5428. 【NOIP2017提高A组集训10.27】查询
- 【JZOJ5428】【NOIP2017提高A组集训10.27】查询
- JZOJ 5428. 【NOIP2017提高A组集训10.27】查询
- 【JZOJ 5428】【NOIP2017提高A组集训10.27】查询
- 【JZOJ 5429】【NOIP2017提高A组集训10.27】排列
- [JZOJ5429]【NOIP2017提高A组集训10.27】排列
- 【JZOJ 5430】【NOIP2017提高A组集训10.27】图
- 【JZOJ5429】【NOIP2017提高A组集训10.27】排列
- 【JZOJ5430】【NOIP2017提高A组集训10.27】图
- 【JZOJ5430】【NOIP2017提高A组集训10.27】图
- jzoj5430 【NOIP2017提高A组集训10.27】图
- 【NOIP2017提高A组集训10.21】Dark
- 【NOIP2017提高A组集训10.21】Fantasy
- 【NOIP2017提高A组集训10.21】 总结
- 【NOIP2017提高A组集训10.21】 总结
- 【NOIP2017提高A组集训10.21】Fantasy
- 【NOIP2017提高A组集训10.24】合影
- 菜鸟详解JDK动态代理之美
- [NOIP模拟] star
- codeforces_884B. Japanese Crosswords Strike Back
- 多彩的Console打印新玩法
- opencv学习——直方图及掩模直方图
- JZOJ5428. 【NOIP2017提高A组集训10.27】查询
- 九周 项 目3 利用二叉树遍历思想解决问题
- nginx
- Linux中的软件管理
- CSS 块元素和行内元素
- 连续第十九天总结
- leetcode 47
- opencv学习——直方图均衡化
- NKOJ 3489 避难向导(LCA+倍增+DFS/DP)