1. 程式人生 > >【mNOIP模擬賽Day 1】 T2 數顏色

【mNOIP模擬賽Day 1】 T2 數顏色

name esp fine www. tar pre ring 由於 使用

題目傳送門:https://www.luogu.org/problemnew/show/P3939

題外話:寫完這題後本地跑了下極限數據,用時1.5s,於是馬上用fread+fwrite優化至0.3s,交至OJ,跑了600+ms,好奇地去掉fread和fwrite,居然只跑了700+ms(總感覺哪裏不太對勁)。

此題有一個性質:對於一個點i,其有且只有一只兔子,且該兔子的顏色是唯一的。考慮到數據範圍較小(兔子數量和顏色數量均$\leq 3*10^{5}$),我們可以對所有顏色開一棵線段樹,維護該區間內有多少只兔子的顏色符合該線段樹要求。更新答案時,先將兩只兔子分別從兩棵線段樹中刪去,然後再插回線段樹中。由於線段樹數量較多,若使用非動態開點線段樹,其空間復雜度為$O(n \times max(c_{i}))$,顯然MLE。故必須采用動態開點線段樹,空間復雜度降低至$O((n+m)\times log(n))$。該做法時間復雜度為$O((n+m)\times log(n))$。

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #define M 310000
 5 using namespace std;
 6 struct sgt{int lc,rc,sum;}a[M*60]={0}; int rt[M]={0},use=0;
 7 int col[M]={0};
 8 int query(int x,int l,int r,int ll,int rr){
 9     if(!x) return 0;
10     if(l<=ll&&rr<=r) return
a[x].sum; 11 int mid=(ll+rr)>>1,sum=0; 12 if(l<=mid) sum=query(a[x].lc,l,r,ll,mid); 13 if(mid<r) sum+=query(a[x].rc,l,r,mid+1,rr); 14 return sum; 15 } 16 int updata(int x,int k,int ll,int rr,int zhi){ 17 if(!x) x=++use; a[x].sum+=zhi; 18 int mid=(ll+rr)>>1
; if(ll==rr) return x; 19 if(k<=mid) a[x].lc=updata(a[x].lc,k,ll,mid,zhi); 20 else a[x].rc=updata(a[x].rc,k,mid+1,rr,zhi); 21 return x; 22 } 23 int main(){ 24 int n,m; scanf("%d%d",&n,&m); 25 for(int i=1;i<=n;i++){ 26 int x; scanf("%d",&x); col[i]=x; 27 rt[x]=updata(rt[x],i,1,n,1); 28 } 29 while(m--){ 30 int op,l,r,x; scanf("%d",&op); 31 if(op==1){ 32 scanf("%d%d%d",&l,&r,&x); 33 printf("%d\n",query(rt[x],l,r,1,n)); 34 }else{ 35 scanf("%d",&x); 36 rt[col[x]]=updata(rt[col[x]],x,1,n,-1); 37 rt[col[x+1]]=updata(rt[col[x+1]],x+1,1,n,-1); 38 rt[col[x]]=updata(rt[col[x]],x+1,1,n,1); 39 rt[col[x+1]]=updata(rt[col[x+1]],x,1,n,1); 40 swap(col[x],col[x+1]); 41 } 42 } 43 }

【mNOIP模擬賽Day 1】 T2 數顏色