题目: 

最长上升子序列问题_#include

 


输入

n=5

a={4,2,3,1,5}

输出

3 (a, a, a,构成的子序列2, 3,5最长)


分析:

这个问题是被称作最长上升子序列( LIS, Longest Increasing Subsequence)的著名问题。这一问

题通过使用DP也能很有效率地求解。我们首先来建立一下递推关系。

定义dp[i]:=以ai为末尾的最长上升子序列的长度

以a结尾的上升子序列是

只包含ai的子序列

在满足j<i并且aj<ai的以aj为结尾的上升子列末尾,追加上ai,后得到的子序列

这二者之一。这样就能得到如下的递推关系:

dp[i]=max {1,dp[j]+1 I j<i且aj<ai}

使用这一递推公式可以在O(n^2 )时间内解决这个问题。

代码如solve1

此外还可以定义其他的递推关系。前面我们利用DP求取针对最末位的元素的最长的子序列。如果子序列的长度相同,那么最末位的元素较小的在之后会更加有优势,所以我们再反过来用DP针对相同长度情况下最小的末尾元素进行求解。

dp[i]:=长度为i+1 的上升子序列中末尾元素的最小值(不存在的话就是INF )

我们来看看如何用DP来更新这个数组。

最开始全部dp[i]的值都初始化为INF。然后由前到后逐个考虑数列的元素,对于每个aj,如果i=0或者dp[i- 1]<aj 的话,就用dp[i]=min(dp[i], aj )进行更新。最终找出使得dp[i]<INF的最大的i+1就是结果了。这个DP直接实现的话,能够与前面的方法一样在O(n^2 )的时间内给出结果,但这一算法还可以进一步优化。 首先dp数列中除INF之外是单调递增的,所以可以知道对于每个am 最多只需要1次更新。对于这次更新究竟应该在什么位置,不必逐个遍历,可以利用二分搜索,这样就可以在O(nlogn)时间内求出结果。

代码如solve2

代码实现:

#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
int n;
const int INF=1e9;
vector<int>a(1002);
vector<int>dp1(1002,1);
vector<int>dp2(1003,INF);
void solve1() {//O(n^2) 
	int ans=0;
	for(int i=0; i<n; ++i) {
		for(int j=0; j<i; ++j)
			if(a[j]<a[i])
				dp1[i]=max(dp1[i],dp1[j]+1);
		ans=max(dp1[i],ans);
	}
	cout<<ans<<endl;
}
void solve2() {//O(nlogn) 
	for(int i=0; i<n; ++i)
		*lower_bound(dp2.begin(),dp2.end(),a[i])=a[i];
	//因为我需要求的是最长子序列长度,而不是哪些元素组成最长子序列;
	//所以上边的更新只影响了最长子序列的构成元素,并没有影响其最长长度 
	cout<<lower_bound(dp2.begin(),dp2.end(),INF)-dp2.begin()<<endl;
}
int main() {
	cin>>n;
	for(int i=0; i<n; ++i)
		cin>>a[i];
	solve1();
	solve2();
	return 0;
}