1996:登山

题目描述

五一到了,PKU-ACM队组织大家去登山观光,队员们发现山上一个有$N$个景点,并且决定按照顺序来浏览这些景点,即每次所浏览景点的编号都要大于前一个浏览景点的编号。同时队员们还有另一个登山习惯,就是不连续浏览海拔相同的两个景点,并且一旦开始下山,就不再向上走了。队员们希望在满足上面条件的同时,尽可能多的浏览景点,你能帮他们找出最多可能浏览的景点数么?

输入格式

第一行:$N(2\leqslant N\leqslant 1000)$景点数
第二行: $N$个整数,每个景点的海拔

输出格式

一行整数,表示最多能浏览的景点数

分析

与合唱队形基本相同,均为最长不下降子序列接最长不上升子序列,但是要求输出结果不同,合唱队形是$N-Ans$,本题只需输出$Ans$即可

状态转移方程

 

 

代码

#include <bits/stdc++.h>

#define Enter puts("")
#define Space putchar(' ')

using namespace std;

typedef long long ll;
typedef double Db;

inline ll Read()
{
    ll Ans = 0;
    char Ch = getchar() , Las = ' ';
    while(!isdigit(Ch))
    {
        Las = Ch;
        Ch = getchar();
    }
    while(isdigit(Ch))
    {
        Ans = (Ans << 3) + (Ans << 1) + Ch - '0';
        Ch = getchar();
    }
    if(Las == '-')
        Ans = -Ans;
    return Ans;
}

inline void Write(ll x)
{
    if(x < 0)
    {
        x = -x;
        putchar('-');
    }
    if(x >= 10)
        Write(x / 10);
    putchar(x % 10 + '0');
}

int T[100001];
int Dp[10][10001];
int Ans;

int main()
{
    int n;
    n = Read();
    for(int i = 1; i <= n; i++)
        T[i] = Read();
    T[0] = 0;
    for(int i = 1; i <= n; i++)
        for(int j = 0; j < i; j++)
            if(T[i] > T[j])
                Dp[1][i] = max(Dp[1][i] , Dp[1][j] + 1);
        T[n + 1] = 0;
        for(int i = n; i > 0; i--)
            for(int j = n + 1; j > i; j--)
                if(T[i] > T[j])
                    Dp[2][i] = max(Dp[2][i] , Dp[2][j] + 1);
        for(int i = 1; i <= n; i++)
            Ans = max(Dp[1][i] + Dp[2][i] - 1, Ans);
        Write(Ans);
        return 0;
}

 

posted @ 2021-05-02 08:12  Tenderfoot  阅读(150)  评论(0编辑  收藏  举报