题目:POJ3903

题意:有一个长为n的数列ai,需要求出这个序列的最长上升子序列的长度。上升子序列指的是对于任意i<j都满足ai<aj的子序列。

这题可以用dp来解

第一种dp方式

定义dp[i]为以 ai 结尾的lis的长度。那么,就有递推公式:

dp[i] = max(dp[i], dp[j]+1)

其中,j满足:j<i且aj< ai

那么,这样dp的时间复杂度为O(N2)

第二种dp方式

定义dp[i]为长度为i+1的lis中,末尾元素的最小值

上述定义是基于贪心的思想的,长度相同的lis,其末尾元素越小,那么可以接上新的元素成为长度+1的lis的可能性更大。因此,我们需要尽可能减小相同长度的lis的末尾元素值。

那么,基于这一假设,先把dp[i]初始化为INF,然后对其进行上述操作,即对于元素a[i],将其更新至第一个末尾元素>=a[i]的lis的末尾,朴素算法的时间复杂度为 O(N2) .

显然,dp数组是单调增加的, 因此,可以使用二分搜索进行优化。这里我们使用algorithm内置的lower_bound函数。

AC代码:

#include <iostream>
#include <string.h>
#include <algorithm>
using namespace std;

#define MAXL 100005
#define INF (1 << 30)

int dp[MAXL];
int a[MAXL];

int main()
{
    ios::sync_with_stdio(false);
    cin.tie(0), cout.tie(0);

    int n;
    while (cin >> n)
    {

        fill(dp, dp + n, INF);

        for (int i = 0; i < n; ++i)
        {
            cin >> a[i];
        }

        for (int i = 0; i < n; ++i)
        {
            *lower_bound(dp, dp + n, a[i]) = a[i]; //dp中第一个大于等于a[i]的,接上a[i]
        }

        int ans = 0;

        ans = lower_bound(dp, dp + n, INF) - dp; //有dp[i+1],则dp[i]不为inf,因此可以找第一个inf的位置,就可以确定最大的lis长度。
        cout << ans << endl;
    }
}

转载请注明来源:https://www.longjin666.top/?p=1105

欢迎关注我的公众号“灯珑”,让我们一起了解更多的事物~

你也可能喜欢

发表评论