标签:结尾 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. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。