Description

花匠栋栋种了一排花,每株花都有自己的高度。花儿越长越大,也越来越挤。栋栋决定把这排中的一部分花移走,将剩下的留在原地,使得剩下的花能有空间长大,同时,栋栋希望剩下的花排列得比较别致。 
具体而言,栋栋的花的高度可以看成一列整数h1, h2, … , hn。设当一部分花被移走后,剩下的花的高度依次为g1, g2, … , gm,则栋栋希望下面两个条件中至少有一个满足: 
条件 A:对于所有的1 ≤ i < m / 2,g_2i > g_2i−1,且g_2i > g_2i+1; 
条件 B:对于所有的1 ≤ i < m / 2,g_2i < g_2i−1,且g_2i < g_2i+1。 
此处2i及2i-1,2i+1都为下标。 
注意上面两个条件在m = 1时同时满足,当m > 1时最多有一个能满足。 
请问,栋栋最多能将多少株花留在原地。

Input

输入的第一行包含一个整数 n,表示开始时花的株数。 
第二行包含 n 个整数,依次为h1, h2,… , hn,表示每株花的高度。

Output

输出一行,包含一个整数 m,表示最多能留在原地的花的株数。

Sample Input


5 3 2 1 2

Sample Output

3

Hint

对于 20%的数据,n ≤ 10; 
对于 30%的数据,n ≤ 25; 
对于 70%的数据,n ≤ 1000,0 ≤ hi ≤ 1000; 
对于 100%的数据,1 ≤ n ≤ 100,000,0 ≤ hi ≤ 1,000,000,所有的h_i随机生成,所有随机数服从某区间内的均匀分布。

题解:
推荐做这道题之前先学会LST的nlogn做法,好了,对于这道题我设了一个状态dp[i][0]表示以a[i]结尾的最长满足条件A的序列长度,dp[i][1]就是以a[i]结尾满足条件b的最长长度,很明显我们可以n平方求出dp[1~n],然后取个max就可以了,转移情况讨论就行了,比如dp[i][0]从前面的dp[j][0]的长度为基数,并且a[j]<a[i]转移,加个1取max,分四种情况就可以了,剩下的自己思考一下。优化就用两颗线段树,分别维护dp[i][0~1]为基数,偶数的情况就nlogn求出来了。
代码:

#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<algorithm>
#include<cstring>
#define inf (1<<30)
const int MAXN=;
using namespace std;
int dp[MAXN][],a[MAXN],n,ans=,shang=,xvv=;
struct tree{
int l,r,max;
}b[*][]; void cl(){
memset(dp,,sizeof(dp));
memset(a,,sizeof(a));
} void build(int xv,int l,int r){
xvv=max(xvv,xv);
if(l==r){
b[xv][].l=l,b[xv][].r=r,b[xv][].max=;
b[xv][].l=l,b[xv][].r=r,b[xv][].max=;
return;
}
b[xv][].l=l,b[xv][].r=r,b[xv][].max=;
b[xv][].l=l,b[xv][].r=r,b[xv][].max=;
int mid=(l+r)/;
build(xv*,l,mid);
build(xv*+,mid+,r);
} void insert(int xv,int aum,int zhi,int x){
int l=b[xv][x].l,r=b[xv][x].r,mid=(l+r)/;
if(l==r&&r==aum){
b[xv][x].max=zhi;
return;
}
if(aum<=mid) insert(xv*,aum,zhi,x);
else insert(xv*+,aum,zhi,x);
b[xv][x].max=max(b[xv*][x].max,b[xv*+][x].max);
} int query(int xv,int l,int r,int x){
int L=b[xv][x].l,R=b[xv][x].r,mid=(L+R)/;
if(L==l&&R==r) return b[xv][x].max;
if(r<=mid) return query(xv*,l,r,x);
else if(l>mid) return query(xv*+,l,r,x);
else return max(query(xv*,l,mid,x),query(xv*+,mid+,r,x));
} void DP(){
dp[][]=dp[][]=;
insert(,a[],dp[][],);
for(int i=;i<=n;i++){
if(a[i]!=){
int x=query(,,a[i]-,);
dp[i][]=max(dp[i][],x+);
}
int xx=query(,a[i]+,shang,);
dp[i][]=max(dp[i][],xx+);
if(dp[i][]%==) insert(,a[i],dp[i][],);
else insert(,a[i],dp[i][],);
}
for(int i=;i<xvv;i++)
b[i][].max=,b[i][].max=;
insert(,a[],dp[][],);
for(int i=;i<=n;i++){
int x=query(,a[i]+,shang,);
dp[i][]=max(dp[i][],x+);
if(a[i]!=){
int xx=query(,,a[i]-,);
dp[i][]=max(dp[i][],xx+);
}
if(dp[i][]%==) insert(,a[i],dp[i][],);
else insert(,a[i],dp[i][],);
}
} int main(){
cl();
scanf("%d",&n);
for(int i=;i<=n;i++) scanf("%d",&a[i]),shang=max(shang,a[i]);
shang++;
build(,,shang);
DP();
for(int i=;i<=n;i++) ans=max(ans,dp[i][]);
for(int i=;i<=n;i++) ans=max(ans,dp[i][]);
printf("%d",ans);
}

【NOIP2013】花匠的更多相关文章

  1. Luogu 1970 NOIP2013 花匠 (贪心)

    Luogu 1970 NOIP2013 花匠 (贪心) Description 花匠栋栋种了一排花,每株花都有自己的高度.花儿越长越大,也越来越挤.栋栋决定把这排中的一部分花移走,将剩下的留在原地,使 ...

  2. NOIP2013 花匠解题报告

    //<NOIP2013> 花匠 /* 最优子结构性质,可以用动规.注意到存在30%的变态数据(1 ≤ n ≤ 100,000, 0 ≤ h_i ≤1,000,000),因此应当找到线性的算 ...

  3. [NOIP2013 花匠] 新人解题报告

    本来按照老师的要求,我学OI的第一份解题报告应是在寒假完成的关于数据结构的基础题,但由于身体原因当时未能完成,那么就在省选赛前临时写几篇吧…… 题目描述 花匠栋栋种了一排花,每株花都有自己的高度.花儿 ...

  4. NOIP2013花匠

    描述 花匠栋栋种了一排花,每株花都有自己的高度.花儿越长越大,也越来越挤.栋栋决定把这排中的一部分花移走,将剩下的留在原地,使得剩下的花能有空间长大,同时,栋栋希望剩下的花排列得比较别致. 具体而言, ...

  5. NOIP2013 花匠

    题目描述 花匠栋栋种了一排花,每株花都有自己的高度.花儿越长越大,也越来越挤.栋栋决定 把这排中的一部分花移走,将剩下的留在原地,使得剩下的花能有空间长大,同时,栋栋希 望剩下的花排列得比较别致. 具 ...

  6. [DP][NOIP2013]花匠

    花匠 问题描述: 花匠栋栋种了一排花,每株花都有自己的高度.花儿越长越大,也越来越挤.栋栋决定把这排中的一部分花移走,将剩下的留在原地,使得剩下的花能有空间长大,同时,栋栋希望剩下的花排列得比较别致. ...

  7. NOIP2013 花匠 DP 线段树优化

    网上一堆题解,我写的是N^2优化的那种,nlogn,O(n)的那种能看懂,但是让我自己在赛场写,肯定没戏了 #include <cstdio> #include <iostream& ...

  8. NOIP2013花匠(波动序列)

    波动序列的定义不用多说,下面给出波动序列的求法. #include<iostream> #include<cstdio> #define N 100002 using name ...

  9. [题解+总结]NOIP动态规划大合集

    1.前言 NOIP2003-2014动态规划题目大合集,有简单的也有难的(对于我这种动态规划盲当然存在难的),今天就把这些东西归纳一下,做一个比较全面的总结,方便对动态规划有一个更深的理解. 2.NO ...

  10. noip2017考前整理(未完)

    快考试了,把我以前写过的题回顾一下.Noip2007 树网的核:floyd,推出性质,暴力.Noip2008 笨小猴:模拟Noip2008 火柴棒等式:枚举Noip2008 传纸条:棋盘dpNoip2 ...

随机推荐

  1. 【Offer】[23] 【链表中环的入口结点】

    题目描述 思路分析 测试用例 Java代码 代码链接 题目描述 一个链表中包含环,如何找出环的入口结点?  思路分析 判断链表中是否有环:用快慢指针的方法,慢指针走一步,快指针走两步,如果快指针追上 ...

  2. postman--请求以及变量设置的实例练习

    我们可以在2个地方添加需要执行的js脚本,一个是Pre-request Script,还有一个tests,我们先看请求之前的 1 在请求被发送到服务器之前:就是在“Pre-request Script ...

  3. Apache RocketMQ 消息队列部署与可视化界面安装

    一.介绍 Apache RocketMQ是一个分布式.队列模型的消息中间件,具有低延迟.高性能和高可靠.万亿级容量和灵活的可扩展性.核心组件由四部分组成:Name Servers,Brokers,Pr ...

  4. 泛型接口、JAVA API、包装类

    泛型接口就是拥有一个或多个类型参数的接口 语法: public interface 接口名<类型形参>{ 方法名(类型形参 类型形参实例); } 示例: public interface ...

  5. Redis常用命令(key、string、List)

    1.Key 1.keys *   查询所有数据 2.exists key名   判断key名是否存在 3.move key名  数据库号(0-15)  移动数据key名到相应的数据库 4.expire ...

  6. Hadoop学习笔记之HBase Shell语法练习

    Hadoop学习笔记之HBase Shell语法练习 作者:hugengyong 下面我们看看HBase Shell的一些基本操作命令,我列出了几个常用的HBase Shell命令,如下: 名称 命令 ...

  7. Python 开发植物大战僵尸游戏

    作者:楷楷 链接:https://segmentfault.com/a/1190000019418065 开发思路 完整项目地址: https://github.com/371854496/pygam ...

  8. Anroid逆向学习从编写so到静动态调试分析arm的一次总结

    Anroid逆向学习从编写so到静动态调试分析arm的一次总结 一.前言 最近跟着教我兄弟学逆向这篇教程学习Android逆向,在第七课后作业反复折腾了好几天,正好在折腾的时候对前面的学习总结一波,动 ...

  9. MySQL设计表规范

    规范总结 所有数据库对象名称必须使用小写字母并用下划线分割 所有数据库对象名称禁止使用 MySQL 保留关键字[设计表后逐一排查] 所有表必须使用 Innodb 存储引擎,数据库和表的字符集统一使用 ...

  10. 从Hybrid到React-Native: JS在移动端的南征北战史

    注:因为不了解Dart,所以本文不对flutter相关内容进行阐述, 实在抱歉 Hybrid Hybird是一种混合开发应用,可以实现JS和Java代码的互通,单纯使用ios/android原生实现, ...