ICode9

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

每日一题——最长递增子序列

2021-03-04 23:36:16  阅读:160  来源: 互联网

标签:信封 子列 nums 递增 mid 数组 序列 长度 最长


前置知识

  1. 最长递增子序列
    给你一个整数数组 nums ,找到其中最长严格递增子序列的长度。
    子序列是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。例如,[3,6,2,7] 是数组 [0,3,1,6,2,2,7] 的子序列。

解答:

维护出现过的所有子列,发现同样长度的子列中只需要维护最后一个值最小的子列(这样的子列后面更可能添加更多值,这里称为最优子列);再进一步,只需要最后一个值;所以只需要维护一个数组d,索引为子列长,值为对应长度最优子列最后一个值。
当长度为i的子列后面可以添加数字,且该数字小于长度+1的尾值时,需要更新长度i+1的尾值。由于数组d是单调上升的,所以很只需要找到最后一个小于该数字的长度,并更新它后面的长度的值。
除了找到最长的子列的情况外,剩余部分只需要二分查找。条件为(d[mid] < nums[i]), 更新pos,并保证区间的收敛即可。

//查找最后一个小于nums[i]的pos
int l = 1, r = len, pos = 0;
while (l <= r) //l == r时也有可能 < nums[i]
  int mid = (l + r) >> 1;
  if(d[mid] < nums[i])
    pos = mid //记录
    l = mid + 1; // 保证收敛
  else
    r = mid - 1; // 保证收敛

二分查找

定界l, r,让l,r依判断条件收敛。
可以直接找到目标,也可以在收敛过程中迭代值。

俄罗斯套娃问题

  1. 俄罗斯套娃信封问题
    给定一些标记了宽度和高度的信封,宽度和高度以整数对形式 (w, h) 出现。当另一个信封的宽度和高度都比这个信封大的时候,这个信封就可以放进另一个信封里,如同俄罗斯套娃一样。
    请计算最多能有多少个信封能组成一组“俄罗斯套娃”信封(即可以把一个信封放到另一个信封里面)。

这个问题解决思路是对w进行排序,从而将问题化为1维。主要是技术性问题

sort + lambda表达式

sort(nums.begin(), nums.end(), [](const auto & e1, const auto & e2){return e1[0] < e2[0] || (e1[0] == e2[0] && e1[1] > e2[1])} );

lower_bound

直接找有序容器第一个大于等于给定下界的元素

auto *it = lower_bound(nums.begin(), nums.end(), num);

顺带一提,upper_bound的作用是找有序容器第一个大于给定上界的元素

标签:信封,子列,nums,递增,mid,数组,序列,长度,最长
来源: https://www.cnblogs.com/laiyk/p/14479849.html

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

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

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

ICode9版权所有