<span>模拟91 题解</span>
A. Dove 打扑克
显然的结论是:超过$\sqrt n$的牌堆不会超过$\sqrt n$个。
所以可以分块维护信息:
对于牌数小于$\sqrt n$的,用一个桶维护。
对于牌数大于$\sqrt n$的,用$vector$暴力插入删除维护。
同时维护答案,为了做到$O(1)$修改,也用分块进行维护。
B. Cicada 与排序
值不同的数,位置一定是固定的。
所以可以只对值相同的数在类似归并的过程中进行$dp$。
考虑每一个值,设$dp_{i,j,dep}$表示这个值在原序列上出现的第$i$个副本,在归并排序深度为$dep$的意义下排名为$j$的概率。
处理出数组$g_{i,j}$表示归并排序过程中,两个指针分别指向$i,j$的概率,
$f_{i,j}$表示第$i$位转移到第$j$位的概率,可以在递推$g$数组的同时简单求出。
然后随便转移就完了。
C. Cicada 拿衣服
考虑题中给出的限制$min+or-max-and$。
不妨分别提出其中的$min-max,or-and$。
可以枚举当前考虑的右端点。
那么对于不同的左端点:
$or-and$只有$2*log max(a_i)$种不同的取值,
因为二进制位分别对应只增不减,只减不增。
$min-max$随着左端点不断减小单调递减。
那么可以枚举每一个不同的取值,这个可以用链表简单维护(其实用两个数组暴力重构就可以)。
二分求出符合要求的最靠左的位置就可以。
然而这样做的复杂度是$O(nlog^2n)$的,其实有一些无用的状态。
因为之要求最靠左的左端点,从左到右枚举左端点所在的区间,在第一个可行区间二分更新答案后结束枚举就完了。
为了二分时$O(1)$查询,可以使用ST表维护。
然而刚刚学到了新的数据结构猫树,而且还可以和$zkw$线段树一起用,所以打起来就很帅。
1 #include<iostream> 2 #include<cstdio> 3 #define lch p<<1 4 #define rch p<<1|1 5 using namespace std; 6 const int N=1e6+50007; 7 struct node{ 8 int pos,And,Or; 9 }sa[105],sb[105]; 10 int n,k,bit,ca,cb; 11 int a[N],mx[22][N],mn[22][N],lg[N],ans[N<<1]; 12 void build(int p,int l,int r,int dep){ 13 if(l==r) return ; 14 int mid=l+r>>1; 15 build(lch,l,mid,dep+1); 16 build(rch,mid+1,r,dep+1); 17 mx[dep][mid]=mn[dep][mid]=a[mid]; 18 mx[dep][mid+1]=mn[dep][mid+1]=a[mid+1]; 19 for(int i=mid-1;i>=l;--i) mn[dep][i]=min(mn[dep][i+1],a[i]),mx[dep][i]=max(mx[dep][i+1],a[i]); 20 for(int i=mid+2;i<=r;++i) mn[dep][i]=min(mn[dep][i-1],a[i]),mx[dep][i]=max(mx[dep][i-1],a[i]); 21 } 22 void dfs(int p,int l,int r,int Ans){ 23 Ans=max(Ans,ans[p]); 24 if(l==r){ 25 if(l>1&&l<=n+1) printf("%d ",Ans); 26 return ; 27 } 28 int mid=l+r>>1; 29 dfs(lch,l,mid,Ans); 30 dfs(rch,mid+1,r,Ans); 31 } 32 inline int query_min(int l,int r){ 33 if(l==r) return a[l]; 34 int k=lg[bit]-lg[(l-1)^(r-1)]; 35 return min(mn[k][l],mn[k][r]); 36 } 37 inline int query_max(int l,int r){ 38 if(l==r) return a[l]; 39 int k=lg[bit]-lg[(l-1)^(r-1)]; 40 return max(mx[k][l],mx[k][r]); 41 } 42 inline void modify(int l,int r,int val){ 43 for(l+=bit-1,r+=bit+1;l^r^1;l>>=1,r>>=1){ 44 if(~l&1) ans[l^1]=max(ans[l^1],val); 45 if(r&1) ans[r^1]=max(ans[r^1],val); 46 } 47 } 48 inline int read(register int x=0,register char ch=getchar(),register int f=0){ 49 for(;!isdigit(ch);ch=getchar()) f=ch=='-'; 50 for(; isdigit(ch);ch=getchar()) x=(x<<1)+(x<<3)+(ch^48); 51 return f?-x:x; 52 } 53 int main(){ 54 n=read(); k=read(); 55 for(int i=1;i<=n;++i) a[i]=read(); 56 for(bit=1;bit<=n+1;bit<<=1); 57 for(int i=1;i<=bit;++i) lg[i]=lg[i>>1]+1; 58 for(int i=1;i<=bit+n;++i) ans[i]=-1; 59 build(1,1,bit,1); 60 for(int i=1;i<=n;++i){ 61 for(int j=1;j<=ca;++j) sa[j].And&=a[i],sa[j].Or|=a[i]; 62 sa[++ca]=(node){i,a[i],a[i]}; cb=0; 63 for(int j=1;j<=ca;++j) if(j==ca||sa[j].And!=sa[j+1].And||sa[j].Or!=sa[j+1].Or) sb[++cb]=sa[j]; 64 for(int j=1;j<=cb;++j) sa[j]=sb[j]; 65 ca=cb; 66 int flag=0; 67 for(int j=1;j<=ca&&!flag;++j) 68 if(sa[j].Or-sa[j].And+query_min(sa[j].pos,i)-query_max(sa[j].pos,i)>=k){ 69 int l=sa[j-1].pos+1,r=sa[j].pos; 70 while(l<r){ 71 int mid=l+r>>1; 72 if(sa[j].Or-sa[j].And+query_min(mid,i)-query_max(mid,i)>=k) r=mid; 73 else l=mid+1; 74 } 75 modify(l,i,i-l+1); 76 flag=1; 77 } 78 } 79 dfs(1,1,bit,-1); 80 return 0; 81 }