题意:有一个长为n的数列,求出这个序列中最长的上升子序列长度(不连续,不能等于)。
解法1:简单dp(n2)
思路:
状态设计:F [ i ] 代表以 A [ i ] 结尾的 LIS 的长度
状态转移:F [ i ] = max { F [ j ] + 1 ,F [ i ] } (1 <= j < i,A[ j ] < A[ i ])
边界处理:F [ i ] = 1 (1 <= i <= n)
1 const int maxn = 103, INF = 0x7f7f7f7f;
2 int a[maxn], f[maxn];
3 int n,ans = -INF;
4 int main()
5 {
6 scanf("%d", &n);
7 for(int i=1; i<=n; i++)
8 {
9 scanf("%d", &a[i]);
10 f[i] = 1;
11 }
12 for(int i=1; i<=n; i++)
13 for(int j=1; j<i; j++)
14 if(a[j] < a[i])
15 f[i] = max(f[i], f[j]+1);
16 for(int i=1; i<=n; i++)
17 ans = max(ans, f[i]);
18 printf("%d\n", ans);
19 return 0;
20 }
解法2:贪心优化(nlogn)
思路:新建一个 low 数组,low [ i ]表示长度为i的LIS结尾元素的最小值。
注意这里的最小值需要更新,而且因为长度为3的LIS结尾元素最小值(low[3])必定大于长度为2的LIS结尾元素最小值(low[4]),所以这个序列就是递增的。
当面对一个数字arr[i],若其大于low[i]数组最后一个值,则增加数组长度并且将其填到最后面。否则更新low[i],更新的时候我们由于单调递增可以二分找位置。
1 /* ********************************************** */
2 int k = 1;
3 dp[k] = arr[1];
4 for(int i = 2; i <= n; i++){
5 if(dp[k] < arr[i]) dp[++k] = arr[i]; //如果比最后一个元素大,那么就添加再最后末尾处
6 else *(lower_bound(dp + 1, dp + 1 + k, arr[i])) = arr[i]; //如果比最后一个元素小,那么就替换该序列第一个比他大的数;
7 }
8 /* ********************************************** */