欢迎访问 生活随笔!

生活随笔

当前位置: 首页 > 编程资源 > 编程问答 >内容正文

编程问答

【bzoj 4059】Non-boring sequences

发布时间:2025/7/14 编程问答 28 豆豆
生活随笔 收集整理的这篇文章主要介绍了 【bzoj 4059】Non-boring sequences 小编觉得挺不错的,现在分享给大家,帮大家做个参考.

这题的重点不在于代码,而在于复杂度分析……

首先我们肯定会写 $n^2$ 暴力,就是每次暴力扫 $[l,r]$ 区间,找到任意一个在此区间中只出现过一次的数。设其下标为 $mid$,显然在这个区间中任取一个子区间,只要这个子区间包含第 $mid$ 个数,这个子区间就是非“无聊的”,所以分治判断 $[l,mid-1]$ 和 $[mid+1,r]$ 两个区间是否不是“无聊的”即可。

接下来考虑优化。相信大家都听说过启发式合并,就是对于两个集合,如果合并这两个区集合的复杂度只与元素数较小的集合的元素数量有关,而与元素数较大的集合无关,设这两个集合的元素数量和为 $n$,那么我们就最多以 $O(n/2)$ 的时间合并这两个集合。

换成启发式合并的术语,就是:对于原序列的每一个位置,当包含这个数的区间往上合并时,只有区间大小至少 $\times 2$ 的情况下这个位置才会造成 $O(1)$ 的时间复杂度,否则这个位置不造成时间复杂度。所以每个位置最多造成 $O(\log{n})$ 的时间复杂度,共有 $n$ 个位置,总时间复杂度为 $O(n\times \log{n})$。

至于每个数是否在某个区间中只出现过一次……预处理一下每个位置的前驱后继即可(即上一个和下一个数值相同的位置),类似于建链表。

1 #include<bits/stdc++.h> 2 #define N 200010 3 using namespace std; 4 inline int read(){ 5 int x=0; bool f=1; char c=getchar(); 6 for(;!isdigit(c);c=getchar()) if(c=='-') f=0; 7 for(; isdigit(c);c=getchar()) x=(x<<3)+(x<<1)+(c^'0'); 8 if(f) return x; 9 return 0-x; 10 } 11 int T,n,a[N],pre[N],nxt[N]; 12 map<int,int> mp; 13 bool check(int l,int r){ 14 if(l>=r) return 1; 15 int i=l, j=r; 16 while(i<=j){ 17 if(pre[i]<l && nxt[i]>r) return check(l,i-1) && check(i+1,r); 18 if(i!=j && pre[j]<l && nxt[j]>r) return check(l,j-1) && check(j+1,r); 19 ++i, --j; 20 } 21 return 0; 22 } 23 int main(){ 24 T=read(); 25 while(T--){ 26 n=read(); 27 memset(nxt,0x7f,sizeof nxt); 28 mp.clear(); 29 map<int,int>::iterator it; 30 for(int i=0;i<n;++i){ 31 a[i]=read(); 32 it=mp.find(a[i]); 33 if(it!=mp.end()) 34 pre[i]=it->second, 35 nxt[it->second]=i; 36 else pre[i]=-1; 37 mp[a[i]]=i; 38 } 39 if(check(0,n-1)) printf("non-"); 40 printf("boring\n"); 41 } 42 return 0; 43 } View Code

upd:建议用 $sort$ 排序建链表,我写个 $map$ 跑了倒数第一……

转载于:https://www.cnblogs.com/scx2015noip-as-php/p/bzoj4059.html

《新程序员》:云原生和全面数字化实践50位技术专家共同创作,文字、视频、音频交互阅读

总结

以上是生活随笔为你收集整理的【bzoj 4059】Non-boring sequences的全部内容,希望文章能够帮你解决所遇到的问题。

如果觉得生活随笔网站内容还不错,欢迎将生活随笔推荐给好友。