题目类型:\(DP\)+\(RMQ\)

传送门:>Here<

题意:给定一个长度为\(N\)的序列,并给出\(M\)次询问。询问区间\([L,R]\)内的最长完美序列。所谓完美序列就是指连续的且内部元素不重复的序列

解题思路

暴力做法:对于每一次询问,用一个队列维护,\(O(n)\)给出答案。因此复杂度是\(O(nm)\)。

考虑能否求出以每个点为结尾的最长完美序列长度\(f[i]\),以及起始位置\(s[i]\)。假设我们已经维护好了一个数组\(lst[i]\)表示数值\(a[i]\)上一次出现的位置。那么\(s[i]\)可以满足转移方程$$s[i]=Max{s[i-1],lst[i]+1}$$其中,\(s[i-1]\)表示以\(i-1\)为结尾的最长完美序列,然而随着元素\(i\)的加入,有可能在\(s[i-1]\)之后存在\(a[i]\),为了避免这种情况,增加\(lst[i]+1\)为底线。

\(lst\)数组并没有简单的方法来维护,那么既然不能\(O(n^2)\),那么开桶好了(记得负数,\(+ad\))

求出\(s[i]\)以后,很明显有$$f[i]=i-s[i]+1$$然后考虑如何应对询问。一个很本能的反应是求出\(Max\{f[l]..f[r]\}\)。然而这不仅慢,也不正确——用\(f[i]\)的前提条件必须是\(s[i] \geq l\)。因此每个点的答案应该是\(Min\{f[i],i-L+1\}\)

然而这样的话复杂度依然是\(O(nm)\),一种思路是进行优化,用\(RMQ\)来\(O(1)\)解答。然而我们没有办法迅速生成所有的\(i-L+1\)。但是不是所有的点都需要考虑\(i-L+1\)?很明显不是,如果一个点的\(s[i] \geq L\)就不需要考虑了。

\(s[i]\)的值有什么特点?容易发现\(s[i]\)是单调不下降的。(不然暴力中,我们是如何用队列进行维护的?)。新加进来的一个点只会导致起始点不动或后移。

因此回到询问中来,所有\(s[i]\geq L\)的点一定都分布在该区间的右侧。因此我们可以二分来找到这个分界点,在分界点的右侧做\(RMQ\)就完全没有任何问题了,我们只需要考虑\(f[i]\)

同样,分布在左侧的节点只需要考虑到\(L\)的距离了。最大的一定是左侧的最右侧的那个点

因此我们先预处理出\(s[i]\),复杂度\(O(n)\)。然后倍增预处理\(O(nlogn)\),对于每次询问需要二分\((mlogn)\)。因此整道题的复杂度是\(O(n+(n+m)logn)\)的

反思

做出本题的关键在于想到维护\(s[i]\),并且看出\(s[i]\)是单调不下降的。而这两点恰好都在暴力中体现了。因此这告诉我们,正解的很多思路都是来源于暴力的。

Code

注意如果二分全都不能满足,那么应该返回\(r+1\)

  1. /*By DennyQi 2018*/
  2. #include <cstdio>
  3. #include <queue>
  4. #include <cstring>
  5. #include <algorithm>
  6. #include <cmath>
  7. using namespace std;
  8. typedef long long ll;
  9. const int MAXN = 200010;
  10. const int AD = 1000000;
  11. const int INF = 1061109567;
  12. inline int Max(const int a, const int b){ return (a > b) ? a : b; }
  13. inline int Min(const int a, const int b){ return (a < b) ? a : b; }
  14. inline int read(){
  15. int x = 0; int w = 1; register char c = getchar();
  16. for(; c ^ '-' && (c < '0' || c > '9'); c = getchar());
  17. if(c == '-') w = -1, c = getchar();
  18. for(; c >= '0' && c <= '9'; c = getchar()) x = (x<<3) + (x<<1) + c - '0'; return x * w;
  19. }
  20. int N,M,l,r,m;
  21. int a[MAXN],s[MAXN],lst[MAXN],b[2*AD+100],f[MAXN],st[MAXN][20];
  22. inline int BinarySearch(int l, int r){
  23. int L = l, R = r, Mid, Ans(r+1);
  24. while(L <= R){
  25. Mid = (L + R) / 2;
  26. if(s[Mid] < l){
  27. L = Mid + 1;
  28. }
  29. else{
  30. R = Mid - 1;
  31. Ans = Mid;
  32. }
  33. }
  34. return Ans;
  35. }
  36. inline int RMQ(int l, int r){
  37. if(l>r) return -1;
  38. int k = log(r-l+1) / log(2);
  39. return Max(st[l][k], st[r-(1<<k)+1][k]);
  40. }
  41. int main(){
  42. // freopen(".in","r",stdin);
  43. N = read(), M = read();
  44. for(int i = 1; i <= N; ++i){
  45. a[i] = read();
  46. lst[i] = b[a[i]+AD];
  47. b[a[i]+AD] = i;
  48. }
  49. s[1] = 1;
  50. for(int i = 2; i <= N; ++i){
  51. s[i] = Max(lst[i]+1, s[i-1]);
  52. f[i] = i-s[i]+1;
  53. st[i][0] = f[i];
  54. }
  55. for(int j = 1; (1<<j) <= N; ++j){
  56. for(int i = 1; i+(1<<j)-1 <= N; ++i){
  57. st[i][j] = Max(st[i][j-1], st[i+(1<<(j-1))][j-1]);
  58. }
  59. }
  60. while(M--){
  61. l = read()+1, r = read()+1;
  62. m = BinarySearch(l,r);
  63. printf("%d\n", Max(m-1-l+1, RMQ(m,r)));
  64. }
  65. return 0;
  66. }

[LOJ10121] 与众不同的更多相关文章

  1. 【LOJ10121】与众不同

    [LOJ10121]与众不同 题面 LOJ 题解 这题是_\(tham\)给\(ztl\)他们做的,然而这道题™居然还想了蛮久... 首先可以尺取出一个位置\(i\)上一个合法的最远位置\(pre_i ...

  2. .NET程序员细数Oracle与众不同的那些奇葩点

    扯淡 距上次接触 Oracle 数据库已经是 N 年前的事了,Oracle 的工作方式以及某些点很特别,那会就感觉,这货就是一个奇葩!最近重拾记忆,一直在折腾 Oracle,因为 Oracle 与众不 ...

  3. InstantClick – 快速响应!让你的网站与众不同

    尽管网络带宽不断增加,但网站并没有更快很多.这是因为最大的瓶颈在于页面加载的延迟.InstantClick 是一个很小的 JavaScript 库,大大加速你的网站响应速度. 在访问者点击一个链接之前 ...

  4. 经典网页设计:20个与众不同的国外 HTML5 网站

    大家都都知道, HTML5 具备所有最新的技术和功能,帮助我们创造平滑过渡,花式图像滑块和动画.如果你正在考虑使用HTML5 来设计自己的网站,那么这个集合能够帮助你. 在过去的10年里,网页设计师使 ...

  5. 与众不同 windows phone 8.0 & 8.1 系列文章索引

    [源码下载] [与众不同 windows phone 7.5 (sdk 7.1) 系列文章索引] 与众不同 windows phone 8.0 & 8.1 系列文章索引 作者:webabcd ...

  6. 与众不同 windows phone (34) - 8.0 新的控件: LongListSelector

    [源码下载] 与众不同 windows phone (34) - 8.0 新的控件: LongListSelector 作者:webabcd 介绍与众不同 windows phone 8.0 之 新的 ...

  7. 与众不同 windows phone (35) - 8.0 新的启动器: ShareMediaTask, SaveAppointmentTask, MapsTask, MapsDirectionsTask, MapDownloaderTask

    [源码下载] 与众不同 windows phone (35) - 8.0 新的启动器: ShareMediaTask, SaveAppointmentTask, MapsTask, MapsDirec ...

  8. 与众不同 windows phone (36) - 8.0 新的瓷贴: FlipTile, CycleTile, IconicTile

    [源码下载] 与众不同 windows phone (36) - 8.0 新的瓷贴: FlipTile, CycleTile, IconicTile 作者:webabcd 介绍与众不同 windows ...

  9. 与众不同 windows phone (37) - 8.0 文件系统: StorageFolder, StorageFile, 通过 Uri 引用文件, 获取 SD 卡中的文件

    [源码下载] 与众不同 windows phone (37) - 8.0 文件系统: StorageFolder, StorageFile, 通过 Uri 引用文件, 获取 SD 卡中的文件 作者:w ...

随机推荐

  1. Redis环境搭建和代码测试及与GIS结合的GEO数据类型预研

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 1.1传统MySQL+ Memcached架构遇到的问题 My ...

  2. gitbook 入门教程之实用插件(新增3个插件)

    插件没有什么逻辑顺序,大家可以按照目录大纲直接定位到感兴趣的插件部分阅读即可. 更多插件正在陆续更新中,敬请期待... 最新更新插件 tbfed-pagefooter 版权页脚插件 gitalk 评论 ...

  3. Swift JSON字符串和字典以及数组的互转

    1.JSONString转换为字典 // JSONString转换为字典 func getDictionaryFromJSONString(jsonString:String) ->NSDict ...

  4. luajit官方性能优化指南和注解

    luajit是目前最快的脚本语言之一,不过深入使用就很快会发现,要把这个语言用到像宣称那样高性能,并不是那么容易.实际使用的时候往往会发现,刚开始写的一些小test case性能非常好,经常毫秒级就算 ...

  5. 【记录】使用在线KMS激活Office系列

    摘要 (有能力的请支持正版office) 网上一些激活工具可能捆绑了木马.病毒.使用激活工具有风险.使用在线KMS来激活则没有这个风险. 找到你的office安装目录 已管理员身份运行cmd输入:cd ...

  6. MFC桌面电子时钟的设计与实现

    目录 核心技术 需求分析 程序设计 程序展示 (一)核心技术 MFC(Micosoft Foundation Class Libay,微基础类库)是微基于Windows平台下的C++类库集合,MFC包 ...

  7. MongoDB的导入与导出

    一.导入与导出可以操作本地的mongodb也可以是远程的mongodb,通用选项: -h host 主机 --port port 端口 -u username 用户名 -p password 密码 如 ...

  8. OKR相关4本书,好书3本

    最近几年看过4本OKR相关的书,有3本是4星.其中第一本是最近看的,剩下3本是2017年看的. OKR源自德鲁克和格鲁夫,跟谷歌是天作之合:4星|<这就是OKR> 4星|<OKR实践 ...

  9. 在Linux系统中同步更新我们的Github博客

    原理介绍 类似于版本管理,我们把我们的hexo博客文件系统在Github上建立一个分支,通过管理分支提交最新的博客文件系统,保证我们博客框架的更新.然后我们基于最新的博客框架,撰写文章,进行Githu ...

  10. idea怎么配置spring

    前提基础: 1.idea软件并JDK成功能用 2.有tacate,并会导入. 3.了解jsp和mvc基本结构 详细介绍: https://www.cnblogs.com/wormday/p/84356 ...