ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

LIS最长上升子序列

2019-08-26 22:05:06  阅读:194  来源: 互联网

标签:结尾 int 最小值 low LIS 序列 长度 最长


题意:有一个长为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  /* ********************************************** */

 

标签:结尾,int,最小值,low,LIS,序列,长度,最长
来源: https://www.cnblogs.com/romaLzhih/p/11415303.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有