★★☆ 输入文件:1flower.in
输出文件:1flower.out
简单对比
时间限制:5 s 内存限制:128 MB
【题目描述】
【输入格式】
【输出格式】
【样例输入】
5 3 5 1 2 2 3 1 1 5 1 2 2 2 2 3 3 5
【样例输出】
2 0 0 1 0 【样例说明】 询问[1, 5]:公主采颜色为1和2的花,由于颜色3的花只有一朵,公主不采;询问[1, 2]:颜色1和颜色2的花均只有一朵,公主不采; 询问[2, 2]:颜色2的花只有一朵,公主不采; 询问[2, 3]:由于颜色2的花有两朵,公主采颜色2的花; 询问[3, 5]:颜色1、2、3的花各一朵,公主不采。
【提示】
【数据范围】
对于100%的数据,1 ≤ n ≤ 10^6,c ≤ n,m ≤10^6。
【来源】
【题目来源】
思路:
1)首先暴力只能够拿到20分
2)莫队(真心不太懂。。。还没写)
3)离线+树状数组维护
上代码:
1)暴力(20)代码
#include <algorithm> #include <iostream> #include <cstring> #include <string> #include <cstdio> using namespace std; const int M = 1e6 + 1; int n,c,m,ans; int color[M],v[M]; inline int reads() { int x=0,f=1;char ch=getchar(); while(ch<'0' || ch>'9') {if(ch=='-') f=-1;ch=getchar();} while(ch>='0' && ch<='9') {x=10*x+ch-'0';ch=getchar();} return x*f; } void orz(int l,int r) { ans=0; memset(v,0,sizeof(v)); for(int j=l;j<=r;j++) v[color[j]]++; for(int j=1;j<=c;j++) if(v[j]>1) ans++; printf("%d\n",ans); } int main() { n=reads();c=reads();m=reads(); for(int i=1;i<=n;i++) color[i]=reads(); for(int i=1,l,r;i<=m;i++) { l=reads();r=reads(); orz(l,r); } return 0; }
2)(伪)正解:
据说是莫队???
为什么我写莫队T了2点qwq
T2点代码qwq:
#include <algorithm> #include <cstring> #include <cstdio> #include <cmath> using namespace std; const int M = 1000100; int n,c,m,size,curL,curR,answer; int a[M],cnt[M],ans[M]; struct G { int l,r,id,block; bool operator < (const G &qwq)const { if(block==qwq.block) return r < qwq.r; return block < qwq.block; } }t[M]; inline int read(int &qwq) { char ch=' ';int q=0,w=1; for(;(ch!='-')&&((ch<'0')||(ch>'9'));ch=getchar()); if(ch=='-') w=-1,ch=getchar(); for(;ch>='0' && ch<='9';ch=getchar())q=q*10+ch-48; qwq=q*w; return qwq; } inline void add(int pre) { cnt[a[pre]]++; if(cnt[a[pre]]==2) answer++; } inline void change(int pre) { cnt[a[pre]]--; if(cnt[a[pre]]==1) answer--; } inline void solve() { curL=1,curR=0; for(int i=1;i<=m;++i) { while(curL<t[i].l) change(curL),curL++; while(curL>t[i].l) curL--,add(curL); while(curR<t[i].r) curR++,add(curR); while(curR>t[i].r) change(curR),curR--; ans[t[i].id]=answer; } } int main() { freopen("1flower.in","r",stdin); freopen("1flower.out","w",stdout); read(n),read(c),read(m); size=sqrt(n); for(int i=1;i<=n;++i) read(a[i]); for(int i=1;i<=m;++i) { read(t[i].l),read(t[i].r); t[i].block=(t[i].l-1)/size+1; t[i].id=i; } sort(t+1,t+m+1); solve(); for(int i=1;i<=m;++i) printf("%d\n",ans[i]); return 0; }
难道是我的读入优化写得太丑了???orz
应该不会!!!
3)正解:
离线+树状数组维护
#include<iostream> #include<cstdio> #include<algorithm> #define Lowbit(x) (x&(-x)) using namespace std; const int M = 1000010; int n; int a[M],gs[M],c[M],next[M],last[M]; int ans[M]; struct node { int l,r,id; bool operator < (const node &qwq)const { return l<qwq.l; } }q[M]; int read() { int x=0,f=1; char ch=getchar(); while(ch<'0' || ch>'9') {if(ch=='-') f=-1;ch=getchar();} while(ch>='0' && ch<='9') {x=x*10+(ch-'0');ch=getchar();} return x*f; } void Update(int k,int x) { while(k<=n) { c[k]+=x; k+=Lowbit(k); } } int Sum(int k) { int sum=0; while(k>0) { sum+=c[k]; k-=Lowbit(k); } return sum; } int main() { freopen("1flower.in","r",stdin); freopen("1flower.out","w",stdout); int c,m,i,L; n=read(),c=read(),m=read(); for(i=1; i<=n; i++) a[i]=read(); for(i=1; i<=m; i++) q[i].l=read(),q[i].r=read(),q[i].id=i; sort(q+1,q+m+1); for(i=1; i<=n; i++) { if(last[a[i]]!=0) next[last[a[i]]]=i; last[a[i]]=i; } for(i=1; i<=n; i++) { gs[a[i]]++; if(gs[a[i]]==2) Update(i,1); } L=1; for(i=1; i<=m; i++) { while(L<q[i].l) { if(next[L]!=0) Update(next[L],-1); if(next[next[L]]!=0) Update(next[next[L]],1); L++; } ans[q[i].id]=Sum(q[i].r)-Sum(q[i].l-1); } for(i=1; i<=m; i++) printf("%d\n",ans[i]); return 0; }
❀❀❀❀❀❀❀❀❀❀❀❀❀❀❀❀❀❀❀❀❀