BZOJ2223 [Coci 2009]PATULJCI 題解&程式碼
阿新 • • 發佈:2019-02-16
題意:給出一個長度為n的序列a滿足1≤a[i]≤n。
又有m組詢問,每次對於一個區間[l,r]問是否存在一個數在[l,r]中出現的次數大於(r-l+1)/2。如果存在,輸出yes,否則輸出no。
題解:標準主席樹…主席樹還是這麼簡單粗暴QwQ但是我MLE了好多次,這空間卡得我懷疑人生…不過最後的錯誤反而不在空間233333
我把n和m弄錯了WA了好幾次,總之是簡單錯誤啦
和BZOJ3524是一道題,連結:
http://blog.csdn.net/rainbow6174/article/details/51074109
/**************************************************************
Problem: 2223
User: Rainbow6174
Language: C++
Result: Accepted
Time:1072 ms
Memory:120412 kb
****************************************************************/
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn = 500005;
const int maxm = 10005;
int n,m,q,l,r,x,cnt,ans,rt[maxn],son[2][maxn*20],sum[maxn*20];
void Insert(int rl,int rn,int l,int r,int v)
{
sum[rn]=sum[rl]+1;
if(l==r)return;
int mid = (l+r)/2 ;
if(mid>=v) son[1][rn]=son[1][rl],son[0][rn]=++cnt,Insert(son[0][rl],son[0][rn],l,mid,v);
else son[0][rn]=son[0][rl],son[1][rn]=++cnt,Insert(son[1][rl],son[1][rn],mid+1,r,v);
//cout<<rl<<' '<<rn<<' '<<sum[son[0][rn]]<<' '<<sum[son[0][rl]]<<' '<<v<<endl;
}
int query(int rl,int rn,int l,int r,int k)
{
if(l==r)return l;
int mid = (l+r)/2;
if(sum[son[0][rn]]-sum[son[0][rl]]>k) return query(son[0][rl],son[0][rn],l,mid,k);
if(sum[son[1][rn]]-sum[son[1][rl]]>k) return query(son[1][rl],son[1][rn],mid+1,r,k);
return 0;
}
int main(void)
{
//freopen("patuljci10.in","r",stdin);
//freopen("patuljci.ans","w",stdout);
scanf("%d%d",&n,&m);
for(int i = 1; i <= n; i++)rt[i]=++cnt;
for(int i = 1; i <= n; i++)
{
scanf("%d",&x);
Insert(rt[i-1],rt[i],1,m,x);
}
scanf("%d",&q);
for(int i = 1; i <= q; i++)
{
scanf("%d%d",&l,&r);
ans=query(rt[l-1],rt[r],1,m,(r-l+1)/2);
if(ans)printf("yes %d\n",ans);
else printf("no\n");
}
return 0;
}