C-勤奋的杨老师 (dp应用+二分)

时间限制:C/C++ 1秒,其他语言2秒
空间限制:C/C++ 32768K,其他语言65536K
64bit IO Format: %lld

题目描述 

杨老师认为他的学习能力曲线是一个拱形。勤奋的他根据时间的先后顺序罗列了一个学习清单,共有n个知识点。但是清单中的知识并不是一定要学习的,可以在不改变先后顺序的情况下有选择的进行学习,而每一个知识点都对应一个难度值。杨老师希望,后学习的知识点的难度一定不低于前一个知识点的难度(i<j时ai<=aj),而可能存在一个临界点,在临界点以后,他希望后学习的知识点的难度一定不高于前一个知识点的难度(i<j时ai>=aj)。杨老师想尽可能多的学习知识。请问:杨老师最多可以学习多少知识?

输入描述:

第一行:一个整数n(0<n<500000)接下来一行:n个整数,第i个整数ai(0<=ai<500000)表示第i道题目的难度。

输出描述:

一行一个整数,表示杨老师最多可以学习多少个知识。

示例1

输入

5
1 4 2 5 1

输出

4

题意:对于某个位置,求前部分的最长上升子序列长度与后部分最长上升子序列长度的和的最大值.

开四个数组,前两个数组分别保存的是前,后部分的最长上升子序列.后两个数组保存的是前,后部分的最长上升子序列长度

最后再暴力走一遍,总的时间复杂度为O(nlogn).

#include<bits/stdc++.h>
using namespace std;
int dp1[500010],dp2[500010],d1[500010],d2[500010],a[500010],n;
void my_solve()
{
    int s=0;
    fill(dp1,dp1+n,INT_MAX);
    fill(dp2,dp2+n,INT_MAX);
    for(int i=0;i<n;i++)
    {
        int k=upper_bound(dp1,dp1+n,a[i])-dp1;
        dp1[k]=a[i];
        d1[i]=k+1;
    }
    for(int i=n-1;i>=0;i--)
    {
        int k=upper_bound(dp2,dp2+n,a[i])-dp2;
        dp2[k]=a[i];
        d2[i]=k+1;
    }
    for(int i=0;i<n;i++)
        s=max(s,d1[i]+d2[i]-1);
    printf("%d\n",s);
}
int main()
{
    scanf("%d",&n);
    for(int i=0;i<n;i++)
        scanf("%d",&a[i]);
    my_solve();
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值