其他分享
首页 > 其他分享> > 每日一题——最长递增子序列

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

作者:互联网

前置知识

  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