[Poi2010]Monotonicity 2

题目

给出N个正整数a[1..N],再给出K个关系符号(>、<或=)s[1..k]。
选出一个长度为L的子序列(不要求连续),要求这个子序列的第i项和第i+1项的的大小关系为s[(i-1)mod K+1]。
求出L的最大值。

INPUT

第一行两个正整数,分别表示N和K (N, K <= 500,000)。
第二行给出N个正整数,第i个正整数表示a[i] (a[i] <= 10^6)。
第三行给出K个空格隔开关系符号(>、<或=),第i个表示s[i]。

OUTPUT

一个正整数,表示L的最大值。

SAMPLE

INPUT

7 3
2 4 3 1 3 5 3
< > =

OUTPUT

6

解题报告

考试时连最最最简单的DP都没想出来,就打了个DFS- -
正解:
我们先考虑只有一种符号的情况,比如说考虑<,那么不就变成了求最长上升子序列吗。
同样的,我们扩展至三种符号:
  1. for(int i=;i<=n;i++){
  2. f[i]=;
  3. for(int j=;j<=i-;j++){
  4. int tmp(f[j]%k+);
  5. if(op[tmp]=='='&&a[j]==a[i]&&f[j]+>f[i])
  6. f[i]=f[j]+;
  7. if(op[tmp]=='>'&&a[j]>a[i]&&f[j]+>f[i])
  8. f[i]=f[j]+;
  9. if(op[tmp]=='<'&&a[j]<a[i]&&f[j]+>f[i])
  10. f[i]=f[j]+;
  11. }
  12. }
这就是最最最简单的DP,然而我们知道,这玩意是O(n²)的复杂度,显然会T,那么我们就需要优化一下了。
我们发现,转移时有O(n)的复杂度来找最大值,那么我们想,是否可以把这个过程优化呢?自然可以,我们的目的在于找到权值符合条件的最大f值,所以,我们需要一个新的东西来完成它:

权值线段树

这是一个神奇的数据结构- -,好吧,也不怎么神奇,它在这道题里是以权值为下标,存入该点最优解的一种线段树,它就可以完成这个伟大的任务啦。
我们需要3棵树(其实2棵也可以,相等的那个用数组模拟即可实现,只是我比较懒- -),每一棵树存以该符号为后面所接符号的权值的最优解(好绕啊- -),这样我们在找的时候,取出每棵树中符合权值条件的最优解,三解进行比较,选出最优以确定符号,继续转移并更新相应的线段树即可。
(我语文表达能力好弱啊)
  1. #include<iostream>
  2. #include<cstring>
  3. #include<cstdio>
  4. using namespace std;
  5. inline int read(){
  6. int sum();
  7. char ch(getchar());
  8. for(;ch<''||ch>'';ch=getchar());
  9. for(;ch>=''&&ch<='';sum=sum*+(ch^),ch=getchar());
  10. return sum;
  11. }
  12. inline char init(){
  13. char ch(getchar());
  14. for(;ch!='='&&ch!='>'&&ch!='<';ch=getchar());
  15. return ch;
  16. }
  17. inline int my_max(int a,int b){
  18. return a>b?a:b;
  19. }
  20. int n,k;
  21. int a[],op[];
  22. int tr_d[],tr_x[],tr_e[];
  23. int ad_d[],ad_x[],ad_e[];
  24. inline void pushup_d(int i){
  25. tr_d[i]=my_max(tr_d[i<<],tr_d[i<<|]);
  26. }
  27. inline void pushup_x(int i){
  28. tr_x[i]=my_max(tr_x[i<<],tr_x[i<<|]);
  29. }
  30. inline void pushup_e(int i){
  31. tr_e[i]=my_max(tr_e[i<<],tr_e[i<<|]);
  32. }
  33. inline void pushdown_d(int i){
  34. if(ad_d[i]){
  35. ad_d[i<<]=ad_d[i];
  36. ad_d[i<<|]=ad_d[i];
  37. tr_d[i<<]=ad_d[i];
  38. tr_d[i<<|]=ad_d[i];
  39. tr_d[i]=ad_d[i];
  40. ad_d[i]=;
  41. }
  42. }
  43. inline void pushdown_x(int i){
  44. if(ad_x[i]){
  45. ad_x[i<<]=ad_x[i];
  46. ad_x[i<<|]=ad_x[i];
  47. tr_x[i<<]=ad_x[i];
  48. tr_x[i<<|]=ad_x[i];
  49. tr_x[i]=ad_x[i];
  50. ad_x[i]=;
  51. }
  52. }
  53. inline void pushdown_e(int i){
  54. if(ad_e[i]){
  55. ad_e[i<<]=ad_e[i];
  56. ad_e[i<<|]=ad_e[i];
  57. tr_e[i<<]=ad_e[i];
  58. tr_e[i<<|]=ad_e[i];
  59. tr_e[i]=ad_e[i];
  60. ad_e[i]=;
  61. }
  62. }
  63. inline void update_d(int ll,int rr,int c,int l,int r,int i){
  64. if(ll<=l&&r<=rr){
  65. ad_d[i]=c;
  66. tr_d[i]=c;
  67. return;
  68. }
  69. pushdown_d(i);
  70. int mid((l+r)>>);
  71. if(ll<=mid)
  72. update_d(ll,rr,c,l,mid,i<<);
  73. if(rr>mid)
  74. update_d(ll,rr,c,mid+,r,i<<|);
  75. pushup_d(i);
  76. }
  77. inline void update_x(int ll,int rr,int c,int l,int r,int i){
  78. if(ll<=l&&r<=rr){
  79. ad_x[i]=c;
  80. tr_x[i]=c;
  81. return;
  82. }
  83. pushdown_x(i);
  84. int mid((l+r)>>);
  85. if(ll<=mid)
  86. update_x(ll,rr,c,l,mid,i<<);
  87. if(rr>mid)
  88. update_x(ll,rr,c,mid+,r,i<<|);
  89. pushup_x(i);
  90. }
  91. inline void update_e(int ll,int rr,int c,int l,int r,int i){
  92. if(ll<=l&&r<=rr){
  93. ad_e[i]=c;
  94. tr_e[i]=c;
  95. return;
  96. }
  97. pushdown_e(i);
  98. int mid((l+r)>>);
  99. if(ll<=mid)
  100. update_e(ll,rr,c,l,mid,i<<);
  101. if(rr>mid)
  102. update_e(ll,rr,c,mid+,r,i<<|);
  103. pushup_e(i);
  104. }
  105. inline int query_d(int ll,int rr,int l,int r,int i){
  106. if(ll>rr)
  107. return ;
  108. if(ll<=l&&r<=rr)
  109. return tr_d[i];
  110. pushdown_d(i);
  111. int mid((l+r)>>);
  112. int ret();
  113. if(ll<=mid)
  114. ret=my_max(ret,query_d(ll,rr,l,mid,i<<));
  115. if(rr>mid)
  116. ret=my_max(ret,query_d(ll,rr,mid+,r,i<<|));
  117. return ret;
  118. }
  119. inline int query_x(int ll,int rr,int l,int r,int i){
  120. if(ll>rr)
  121. return ;
  122. if(ll<=l&&r<=rr)
  123. return tr_x[i];
  124. pushdown_x(i);
  125. int mid((l+r)>>);
  126. int ret();
  127. if(ll<=mid)
  128. ret=my_max(ret,query_x(ll,rr,l,mid,i<<));
  129. if(rr>mid)
  130. ret=my_max(ret,query_x(ll,rr,mid+,r,i<<|));
  131. return ret;
  132. }
  133. inline int query_e(int ll,int rr,int l,int r,int i){
  134. if(ll>rr)
  135. return ;
  136. if(ll<=l&&r<=rr)
  137. return tr_e[i];
  138. pushdown_e(i);
  139. int mid((l+r)>>);
  140. int ret();
  141. if(ll<=mid)
  142. ret=my_max(ret,query_e(ll,rr,l,mid,i<<));
  143. if(rr>mid)
  144. ret=my_max(ret,query_e(ll,rr,mid+,r,i<<|));
  145. return ret;
  146. }
  147. int f[];
  148. int mx();
  149. int main(){
  150. n=read(),k=read();
  151. for(int i=;i<=n;i++)
  152. a[i]=read(),mx=my_max(mx,a[i]);
  153. for(int i=;i<=k;i++){
  154. char ch(init());
  155. if(ch=='>')
  156. op[i]=;
  157. if(ch=='<')
  158. op[i]=;
  159. if(ch=='=')
  160. op[i]=;
  161. }
  162. f[]=;
  163. if(op[]==)
  164. update_d(a[],a[],f[],,mx,);
  165. if(op[]==)
  166. update_x(a[],a[],f[],,mx,);
  167. if(op[]==)
  168. update_e(a[],a[],f[],,mx,);
  169. for(int i=;i<=n;i++){
  170. int now(a[i]);
  171. int ans_d(query_d(now+,mx,,mx,));
  172. int ans_x(query_x(,now-,,mx,));
  173. int ans_e(query_e(now,now,,mx,));
  174. int ans(my_max(my_max(ans_d,ans_x),ans_e));
  175. f[i]=ans+;
  176. int o(op[ans%k+]);//cout<<i<<' '<<f[i]<<' '<<o<<endl;
  177. if(o==)
  178. update_d(now,now,f[i],,mx,);
  179. if(o==)
  180. update_x(now,now,f[i],,mx,);
  181. if(o==)
  182. update_e(now,now,f[i],,mx,);
  183. }
  184. int mxx();
  185. for(int i=;i<=n;i++)
  186. mxx=my_max(mxx,f[i]);
  187. printf("%d",mxx);
  188. }
写的极其丑- -,毕竟三颗线段树乱搞
凑合着看吧,其实理解了之后,一颗线段树,加不同的域,对传的参数进行处理,就可以达到三颗线段树的效果

[补档][Poi2010]Monotonicity 2的更多相关文章

  1. [补档]暑假集训D3总结

    考试 集训第一次考试,然而- -   总共四道题,两道打了DFS,一道暴力,一道~~输出样例~~乱搞,都是泪啊- - 目前只改了三道,回头改完那道题再上题解吧- - T2 [Poi2010]Monot ...

  2. BZOJ2090: [Poi2010]Monotonicity 2【线段树优化DP】

    BZOJ2090: [Poi2010]Monotonicity 2[线段树优化DP] Description 给出N个正整数a[1..N],再给出K个关系符号(>.<或=)s[1..k]. ...

  3. 【BZOJ2090/2089】[Poi2010]Monotonicity 2 动态规划+线段树

    [BZOJ2090/2089][Poi2010]Monotonicity Description 给出N个正整数a[1..N],再给出K个关系符号(>.<或=)s[1..k].选出一个长度 ...

  4. STL 补档

    STL 补档 1.vector 作用:它能够像容器一样存放各种类型的对象,简单地说,vector是一个能够存放任意类型的动态数组,能够增加和压缩数据. vector在C++标准模板库中的部分内容,它是 ...

  5. 图论补档——KM算法+稳定婚姻问题

    突然发现考前复习图论的时候直接把 KM 和 稳定婚姻 给跳了--emmm 结果现在刷训练指南就疯狂补档.QAQ. KM算法--二分图最大带权匹配 提出问题 (不严谨定义,理解即可) 二分图 定义:将点 ...

  6. [补档] 大假期集训Part.1

    新博客搭起来先补一发档... 那就从大假期集训第一部分说起好了QwQ 自己还是太菜掉回了2016级水平 day1: day1的时候来得有点晚(毕竟准高一)然后进机房发现早就开考了还没有给我题面于是搞了 ...

  7. 软件安装配置笔记(三)——ArcGIS系列产品安装与配置(补档)(附数据库连接及数据导入)

    在前两篇安装配置笔记之后,就忘记把其他安装配置笔记迁移过来了,真是失误失误!趁现在其他文档需要赶紧补上. 目录: 一.ArcMap 二.ArcMap连接数据库并导入数据 三.Arcgis Pro 四. ...

  8. 补档 Codeblocks下的文件标题栏(标签)显示方法

    可能在以下链接也能看到这篇文档 我知道很多人都不知道这个到底叫啥,还不如直接一点: 文件标题栏 就是如下的效果. 解决办法: 在左上角第三个view下,打开后取消Hide editor tabs 选项 ...

  9. [BZOJ2090/2089] [Poi2010]Monotonicity 2/Monotonicity 树状数组优化dp

    这个dp乍看不科学,仔细一看更不科学,所以作为一个执着BOY,我决定要造数据卡死波兰人民,但是我造着造着就......证出来了......... 这个就是把 < > =分开讨论每次找到f[ ...

随机推荐

  1. linux中常用的命令

    1.向某个ip发送文件 scp name.tar root(身份)@ip:/lujing 2.重启系统 init 6 3.如果修改了ifcfg-eth0类似于网卡配置文件,修改网口,ip等设置: 需要 ...

  2. salesforce零基础学习(七十四)apex:actionRegion以及apex:actionSupport浅谈

    我们在开发中,很难会遇见不提交表单的情况.常用的apex:commandButton,apex:commandLink,apex:actionFunction,apex:actionSupport.他 ...

  3. Jquery Ajax 保存

    Jquery Ajax 保存: $.ajax({ type: "POST", async:false, url: "${ctx}/url", data: {pI ...

  4. 【LeetCode】89. Gray Code

    题目: The gray code is a binary numeral system where two successive values differ in only one bit. Giv ...

  5. 【Android Developers Training】 86. 基于连接类型修改您的下载模式

    注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...

  6. JavaScript中的排序

    <script> //1. 冒泡排序 function bubbleSort(arr) { var len = arr.length; for (var i = 0; i < len ...

  7. 什么是VPN,VPN有什么用,怎么获得VPN

    什么是VPN? VPN英文全称是“Virtual Private Network”,翻译过来就是“虚拟专用网络”.vpn被定义为通过一个公用网络(通常是因特网)建立一个临时的.安全的连接,是一条穿过混 ...

  8. JUnit【1】断言用法之assertEquals/True/False/ArrayEquals

    前段时间去亚信面试,被问到写一个冒泡排序,心想这多新鲜,刷刷几下写好.面试官突然问,你怎么对这个程序进行单元测试?    单元测试?!    懵圈...      单元测试      代码是为了什么, ...

  9. spring auto-config

    spring security auto-config auto-config配置 <http auto-config="true"> </http> 自动 ...

  10. java项目(非ssm等框架)下的quartz定时器任务

    第一步:引包 要使用Quartz,必须要引入以下这几个包: 1.log4j-1.2.16 2.quartz-2.1.7 3.slf4j-api-1.6.1.jar 4.slf4j-log4j12-1. ...