牛客挑战赛36-C 纸飞机 //dp优化

17 篇文章 1 订阅
1 篇文章 0 订阅

题面链接

https://ac.nowcoder.com/acm/contest/3782/C

题目大意

对于原数组的每个数 a [ i ] a[i] a[i],分别求出去除 a [ i ] a[i] a[i]后,能覆盖另外 n − 1 n-1 n1个数的最少递减子序列的个数。

思路

首先,去除 a [ i ] a[i] a[i]后,需要用最少的递减子序列覆盖另外 n − 1 n-1 n1个数,也就是求:去除 a [ i ] a[i] a[i]后的最小链覆盖。根据结论最小链覆盖等于最长反链,也就是求:去除 a [ i ] a[i] a[i]后,序列的最长不下降子序列。
对于原序列的最长非下降子序列,很容易求解:定义 d p [ i ] dp[i] dp[i]为以 a [ i ] a[i] a[i]结尾的最长不下降子序列长度,状态转移方程为: d p [ i ] = m a x ( d p [ j ] ) + 1 ( a [ j ] < = a [ i ] 且 j < i ) dp[i]=max(dp[j])+1(a[j]<=a[i]且j<i) dp[i]=max(dp[j])+1(a[j]<=a[i]j<i),由于 n n n范围较大,可以用树状数组/线段树维护前缀最大值,实现 O ( n l o g n ) O(nlog_n) O(nlogn)复杂度求解原数组每个数的最长不下降子序列长度,由于 a [ i ] a[i] a[i]范围较大,需要先对原数组进行离散化,才好用数据结构维护。
设求出的原序列最长不下降子序列长度为 a n s ans ans,对于每个数 a [ i ] a[i] a[i],去除之后,序列的最长不下降子序列长度和" a [ i ] a[i] a[i]是否为原序列最长不下降子序列的一员且无法被取代"有关。如果是且无法被取代,则结果为 a n s − 1 ans-1 ans1;否则为 a n s ans ans
用一个辅助数组 m a ma ma,定义 m a [ i ] ma[i] ma[i]为放在原序列最长非递减子序列第 i i i位的最大值,将 m a [ a n s + 1 ] ma[ans+1] ma[ans+1]设为无穷大,方便统一处理。再用一个 v i s vis vis数组,记录第 i i i个数是否可以成为原序列最长非递减子序列的一员。然后从后往前维护这两个数组:如果 m a [ d p [ i ] + 1 ] > = a [ i ] ma[dp[i]+1]>=a[i] ma[dp[i]+1]>=a[i],说明 a [ i ] a[i] a[i]可以是原序列最长非递减子序列的第 i i i位,也就是 a [ i ] a[i] a[i]可以成为原序列最长非递减子序列的一员,但是否为原序列最长非递减子序列第 i i i位的最大值需要判断一下来更新维护。
之后,再用一个 s u m sum sum数组,来记录有多少个数可以是原序列最长非递减子序列的第 i i i位,目的是:判断 a [ i ] a[i] a[i]能否被取代,当 v i s [ i ] 且 s u m [ d p [ i ] ] = 1 vis[i]且sum[dp[i]]=1 vis[i]sum[dp[i]]=1时,说明 a [ i ] a[i] a[i]可以是原序列最长非递减子序列的一员,且不能被取代。否则说明可以被取代。

参考代码

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#include <map>
using namespace std;
typedef long long ll;
typedef pair<int,int> P;
const int maxn=1e6+7;
const int inf=0x3f3f3f3f;
#define pb push_back
#define ft first
#define sd second
#define ms(x,y) memset(x,y,sizeof(x))
int n,ans;
int a[maxn],b[maxn],dp[maxn];
int ma[maxn],sum[maxn];
bool vis[maxn];
struct BIT{
    int bit[maxn];
    inline int lowbit(int x){return x&(-x);}
    inline void init(){ for(int i=1;i<=n;i++) bit[i]=0; }
    inline void update(int x,int v){for(int i=x;i<=n;i+=lowbit(i))bit[i]=max(bit[i],v);}
    inline int query(int x,int ans=0){for(int i=x;i;i-=lowbit(i))ans=max(bit[i],ans);return ans;}
}s;
void lis(){//离散化
    sort(b+1,b+1+n);
    int len=unique(b+1,b+1+n)-b-1;
    for(int i=1;i<=n;i++){
        a[i]=lower_bound(b+1,b+1+len,a[i])-b;
    }
}
int main()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++){
        scanf("%d",&a[i]);
        b[i]=a[i];
    }
    lis();
    s.init();
    for(int i=1;i<=n;i++){
        dp[i]=s.query(a[i])+1;
        s.update(a[i],dp[i]);
        ans=max(ans,dp[i]);
    }
    ma[ans+1]=inf;
    for(int i=n;i>=1;i--){
        if(ma[dp[i]+1]>=a[i]){//a[i]能否放在原序列最长非下降子序列第dp[i]位
            ma[dp[i]]=max(ma[dp[i]],a[i]);//a[i]能否成为原序列最长非下降子序列第dp[i]位的最大值
            vis[i]=true;//标记一下
        }
    }
    for(int i=1;i<=n;i++)if(vis[i])sum[dp[i]]++;//统计有多少个数可以是原序列最长非递减子序列的第$i$位
    for(int i=1;i<=n;i++){
        printf("%d%c",(vis[i]&&sum[dp[i]]==1)?ans-1:ans,i==n?'\n':' ');
    }
    return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值