[Poi2010]Monotonicity 2 线段树
这道题考试的时候先打了个dfs暴力。又打了个O(n²)的动规。然后竟然心血来潮拍了一下。。明明知道过不去的。。。然后水了50分(20个测试点这么多啊啊啊啊)。
因为它已经提前给你如果长度为i时下一位的符号,显然O(n²)的打法大家都会打。
for(i..1..n)
for(j..1..i-1)
if(符合条件)
f[i]=max(f[i],f[j]+1);
然后我们就考虑如何降低复杂度。
因为我们要求的是符合条件的前缀最大值,可以使用线段树来维护。
线段树的下标是权值,内容是最长长度。
一共三种情况:<,>,=。
我们需要单独存这三种情况。以大于为例:
线段树里面存的是某个权值下,下一位是大于号的所有最优长度。
至于等于可以用数组记录,不必用线段树。
然后我们更新的时候,对于f[i]=max{权值(0,a[i]-1)中所有后边符号是小于的最大值,权值(a[i]+1,inf)中所有后边符号是大于的最大值,相等的最大值}+1;
再根据f[i]的长度确定后边一位是什么符号,再把f[i]加入对应的线段树中。
复杂度O(n㏒n)。
#include<cstdio>
#include<cstring>
#include<iostream>
using namespace std;
#define pos(i,a,b) for(int i=(a);i<=(b);i++)
#define N 2020200
int a[N],n,k;
char s[N];
int cun[N],f[N];
int ans;
int ma;
struct haha{
int left,right,num;
};
haha treeda[N*4];haha treexiao[N*4];
void buildda(int x,int left,int right){
treeda[x].left=left;
treeda[x].right=right;
if(left==right){
treeda[x].num=0;
return;
}
int mid=(left+right)>>1;
buildda(x*2,left,mid);
buildda(x*2+1,mid+1,right);
}
int queryda(int left,int right,int root){
if(treeda[root].left==left&&treeda[root].right==right){
return treeda[root].num;
}
int mid=(treeda[root].left+treeda[root].right)>>1;
if(right<=mid){
return queryda(left,right,root*2);
}
else{
if(left>mid){
return queryda(left,right,root*2+1);
}
else{
return max(queryda(left,mid,root*2),queryda(mid+1,right,root*2+1));
}
}
}
void changeda(int x,int num,int root){
if(treeda[root].left==treeda[root].right){
treeda[root].num=num;
return;
}
treeda[root].num=max(treeda[root].num,num);
if(x<=treeda[root*2].right)
changeda(x,num,root*2);
else
changeda(x,num,root*2+1);
}
void buildxiao(int x,int left,int right){
treexiao[x].left=left;
treexiao[x].right=right;
if(left==right){
treexiao[x].num=0;
return;
}
int mid=(left+right)>>1;
buildxiao(x*2,left,mid);
buildxiao(x*2+1,mid+1,right);
}
int queryxiao(int left,int right,int root){
if(treexiao[root].left==left&&treexiao[root].right==right){
return treexiao[root].num;
}
int mid=(treexiao[root].left+treexiao[root].right)>>1;
if(right<=mid){
return queryxiao(left,right,root*2);
}
else{
if(left>mid){
return queryxiao(left,right,root*2+1);
}
else{
return max(queryxiao(left,mid,root*2),queryxiao(mid+1,right,root*2+1));
}
}
}
void changexiao(int x,int num,int root){
if(treexiao[root].left==treexiao[root].right){
treexiao[root].num=num;
return;
}
treexiao[root].num=max(treexiao[root].num,num);
if(x<=treexiao[root*2].right)
changexiao(x,num,root*2);
else
changexiao(x,num,root*2+1);
}
int deng[N];
int Max(int a,int b,int c){
return max(max(a,b),c);
}
int main(){
//freopen("mot.in","r",stdin);
//freopen("mot.out","w",stdout);
scanf("%d%d",&n,&k);
pos(i,1,n){
scanf("%d",&a[i]);
ma=max(ma,a[i]);
}
pos(i,1,k){
scanf("%s",&s[i]);
}
pos(i,1,n-1){
cun[i]=s[(i-1)%k+1];
}
buildxiao(1,0,ma);
buildda(1,0,ma);
pos(i,1,n){
int tmp1,tmp2;
if(a[i]==ma){
tmp1=1;
}
else{
tmp1=queryda(a[i]+1,ma,1);
}
if(a[i]==0){
tmp2=1;
}
else{
tmp2=queryxiao(0,a[i]-1,1);
}
f[i]=Max(tmp1,tmp2,deng[a[i]])+1;
if(cun[f[i]]=='>'){
changeda(a[i],f[i],1);
}
if(cun[f[i]]=='<'){
changexiao(a[i],f[i],1);
}
if(cun[f[i]]=='='){
deng[a[i]]=max(deng[a[i]],f[i]);
}
ans=max(ans,f[i]);
}
printf("%d",ans);
return 0;
}
[Poi2010]Monotonicity 2 线段树的更多相关文章
- BZOJ2090: [Poi2010]Monotonicity 2【线段树优化DP】
BZOJ2090: [Poi2010]Monotonicity 2[线段树优化DP] Description 给出N个正整数a[1..N],再给出K个关系符号(>.<或=)s[1..k]. ...
- 【BZOJ2090/2089】[Poi2010]Monotonicity 2 动态规划+线段树
[BZOJ2090/2089][Poi2010]Monotonicity Description 给出N个正整数a[1..N],再给出K个关系符号(>.<或=)s[1..k].选出一个长度 ...
- [补档][Poi2010]Monotonicity 2
[Poi2010]Monotonicity 2 题目 给出N个正整数a[1..N],再给出K个关系符号(>.<或=)s[1..k]. 选出一个长度为L的子序列(不要求连续),要求这个子序列 ...
- bzoj3932--可持久化线段树
题目大意: 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的 任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第 ...
- codevs 1082 线段树练习 3(区间维护)
codevs 1082 线段树练习 3 时间限制: 3 s 空间限制: 128000 KB 题目等级 : 大师 Master 题目描述 Description 给你N个数,有两种操作: 1:给区 ...
- codevs 1576 最长上升子序列的线段树优化
题目:codevs 1576 最长严格上升子序列 链接:http://codevs.cn/problem/1576/ 优化的地方是 1到i-1 中最大的 f[j]值,并且A[j]<A[i] .根 ...
- codevs 1080 线段树点修改
先来介绍一下线段树. 线段树是一个把线段,或者说一个区间储存在二叉树中.如图所示的就是一棵线段树,它维护一个区间的和. 蓝色数字的是线段树的节点在数组中的位置,它表示的区间已经在图上标出,它的值就是这 ...
- codevs 1082 线段树区间求和
codevs 1082 线段树练习3 链接:http://codevs.cn/problem/1082/ sumv是维护求和的线段树,addv是标记这歌节点所在区间还需要加上的值. 我的线段树写法在运 ...
- PYOJ 44. 【HNSDFZ2016 #6】可持久化线段树
#44. [HNSDFZ2016 #6]可持久化线段树 统计 描述 提交 自定义测试 题目描述 现有一序列 AA.您需要写一棵可持久化线段树,以实现如下操作: A v p x:对于版本v的序列,给 A ...
随机推荐
- Failed to sync Gradle project 'XX'错误解决
错误代码 Failed to sync Gradle project 'WeChat' Error:Failed to find target with hash string 'android-24 ...
- JavaScript 语言中的 this
JavaScript 语言中的 this 由于其运行期绑定的特性,JavaScript 中的 this 含义要丰富得多,它可以是全局对象.当前对象或者任意对象,这完全取决于函数的调用方式.JavaSc ...
- spring +springmvc+mybatis组合applicationContext.xml文件配置
<?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.spr ...
- 轻量级代码生成器-OnlyCoder 第二篇
最近利用业余时间将OnlyCoder又重新打造了一番,使其使用起来更简单.更顺手. 相关的帮助文档也已发布了版本. 界面改版,UI采用了DotNetBar2组件. 还是先看下UI效果吧. 由于使用了 ...
- git分支的使用
本文章假定你已经接触了一些git的基本概念和基本的操作知识 这里先贴出关于分支的一些常用命令 git branch /*查看所有分支*/git branch <branch-name> / ...
- [CF337D]邪恶古籍-树状dp
Problem 邪恶古籍 题目大意 给出一些关键点,求这棵树上到最远关键点距离小于等于d的有多少个. Solution 一个非常简单的树形dp.然而我被这道题给玩坏了. 在经过分析以后,我们发现只需要 ...
- Redis 错误1067:进程意外终止,Redis不能启动,Redis启动不了
Redis 错误1067:进程意外终止,Redis不能启动,Redis启动不了 >>>>>>>>>>>>>>> ...
- pc端的企业网站(IT修真院test8)详解1-2
今天接着说test8的页面还原. 头部header和尾部footer 我一开始是想直接使用bootstrap的栅格系统来实现的.但是内容的定位出了不少麻烦. 那么就索性用原生html+css来还原页面 ...
- Excel导出百万级数据解决方案
因项目业务,需要导出百万级数据到excel,在研究了各种方案后,最终确定了用POI的SXSSFWorkbook. SXSSFWorkbook是POI3.8以上新增的,excel2007后每个sheet ...
- STK卫星工具箱下载
简介 STK的全称是Satellite Tool Kit(卫星工具箱),STK/Pro 9.0最新出品,完整版,是由Analytical Graphics公司开发的一款在航天工业领域中处于绝对领先地位 ...