新疆大学ACM-ICPC程序设计竞赛五月月赛(同步赛)C 勤奋的杨老师【DP/正反LIS/类似合唱队形】...

本文介绍了一种通过双向最长递增子序列(LIS)算法来解决的问题,即如何帮助杨老师根据难度值选择尽可能多的知识点进行学习。该问题允许知识点难度在某个点之前递增(可相等),之后递减(也可相等)。通过使用upper_bound方法实现不严格单调条件。
链接:https://www.nowcoder.com/acm/contest/116/C
来源:牛客网

题目描述 
杨老师认为他的学习能力曲线是一个拱形。勤奋的他根据时间的先后顺序罗列了一个学习清单,共有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

【分析】:类似《合唱队形》,双向LIS,注意不严格单调(即可以相等)用upper_bound!

#include<bits/stdc++.h>
using namespace std;
const int N=500005;
int b[N],le[N],ri[N],n,ans=0x7fffffff;
void lis1(){
    int f[N],len=1;
    f[1]=b[1];
    le[1]=1;
    for(int i=2;i<=n;i++){
        if(f[len]<=b[i]) f[++len]=b[i];
        else f[upper_bound(f+1,f+len+1,b[i])-f]=b[i];
        le[i]=len;
    }
}
void lis2(){
    int f[N], len=1;
    f[1]=b[n];
    for(int i=n-1;i>=1;i--){
        if(f[len]<=b[i]) f[++len]=b[i];
        else f[upper_bound(f+1,f+len+1,b[i])-f]=b[i];
        ri[i]=len;
    }
}
int main(){
    cin>>n;
    for(int i=1;i<=n;i++) cin>>b[i];
    lis1();lis2();
    for(int i=1;i<=n;i++)
        ans=min(ans,n-le[i]-ri[i+1]);
    cout<<n-ans;
}

转载于:https://www.cnblogs.com/Roni-i/p/8992131.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值