最长递增子序列问题(你真的会了吗)

这篇具有很好参考价值的文章主要介绍了最长递增子序列问题(你真的会了吗)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

目录

一.最长递增子序列问题I

二.最长递增子序列问题II

三. 最长递增子序列问题III


一.最长递增子序列问题I

1.对应牛客网链接

最长上升子序列(一)_牛客题霸_牛客网 (nowcoder.com)

2.题目描述:

最长递增子序列,算法,leetcode

 3.解题思路

1.首先我们分析题意:最长递增子序列拆:要递增的,还是序列,不一定连续 ,要长度最长的。

2.子序列和子数组问题我们一般考虑必须以某个位置结尾如何如何,在本题中我们可以这样考虑必须以i位置结尾的情况下最长递增子序列的最大长度是多少我们每个位置都这么干那么答案一定就在其中

下面以[5,7,1,9,4,6,2,8,3]为例:

最长递增子序列,算法,leetcode

第一个元素 5: 递增长度只能为1, 接下来第二个7,比5大,长度为2

最长递增子序列,算法,leetcode

 

第三个元素 1:前面没有比它大的,只能为1 , 第四个元素9,前面有5,7,故长度为3到这里你发现,9比7大,7往前构成的长度是2,那9就可以接在7的后面,变成长度加一的新序列。

最长递增子序列,算法,leetcode

最长递增子序列,算法,leetcode

 我相信老铁应该懂了

4.对应代码:

class Solution {
  public:
    /**
     * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
     *
     * 给定数组的最长严格上升子序列的长度。
     * @param arr int整型vector 给定的数组
     * @return int整型
     */
    int LIS(vector<int>& arr) {
        if (arr.empty())
            return 0;
        // write code here
        int N = arr.size();
        vector<int>dp(N, 1);//每个位置最长递增子序列的长度至少为1自己本身就是
        //dp的含义是必须以i位置结尾的情况下最长递增子序列的最大长度
        int maxLen = 1;
        for (int i = 1; i < N; i++) {
            for (int j = 0; j < i; j++) {
                if (arr[i] > arr[j]) {
                    dp[i] = max(dp[i], dp[j] + 1); //长度加1
                }
            }
            maxLen = max(maxLen, dp[i]);//更新最大长度
        }
        return maxLen;
    }

};

二.最长递增子序列问题II

1.对应牛客网链接:

最长上升子序列(二)_牛客题霸_牛客网 (nowcoder.com)

2.题目描述:

最长递增子序列,算法,leetcode

 3.解题思路:

1.在这里我们引入end数组,end[i]的值为最目前为止长度为i+1的最小结尾

2.每次去end数组里面去找大于等于arr[i]最左的位置

最长递增子序列,算法,leetcode

 

我们看同样是长度为2的子序列,[2,3]就比[2,5]好。因为[2,3]后面如果有4的话,组成[2,3,4]长度就是3了,但是[2,5]因为不满足条件,就没法组队了。

我们组成子序列的时候,不仅要让这个序列尽可能的长,而且要让子序列中的上升的时候尽可能的缓慢,[2,3]就比[2,5]上升的缓慢,这样就有机会能拼接出更长的上升子序列。我们用一个数组来保存当前的最长上升子序列,这个数组是严格递增的。
因为是严格递增的,数组中最后一个值nums[max]就是最大值,如果下次再碰到一个数字n,它比num[max]还要大,那么很明显,这个子序列的长度就要+1,并且将数组n添加到数组的末尾。

[2,3,7,8,11,13,18]是目前为止最长的上升子序列,之后如果又碰到了19,或者101,因为他们都大于数组中的最大值18,所以直接将其添加到数组末尾就可以了,同时子序列的长度要+1。19和101的例子很好理解,但如果下次碰到的数字是6或者12呢?因为要让子序列上升的尽可能缓慢,那么让[2,5,7...]变成[2,5,6...]更合适,因为后者上升的更缓慢。同样,将[...8,11,13,18]变成[...8,11,12,18]也是上升的更缓慢一点。
也就是,已知上升子序列[i,i_1,i_2,....,i_n],现在我们在继续遍历的过程中碰到了一个值i_k,这个值是小于i_n的,所以上升子序列的长度还是不变。但是我们需要找到一个位置,将i_k替换掉某个旧的值。

对应动图:

最长递增子序列,算法,leetcode

4.对应代码:

class Solution {
  public:
    /**
     * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
     *
     * 该数组最长严格上升子序列的长度
     * @param a int整型vector 给定的数组
     * @return int整型
     */
    int LIS(vector<int>& arr) {
        // write code here
        if (arr.empty()) {
            return 0;
        }
        int N = arr.size();
        vector<int>end(N);
        //end[i]的含义是目前为止长度为i+1的最小结尾
        end[0] = arr[0];
        int L = 0;
        int maxLen = 1;
        int right = 0; //用于控制end数组的范围
        int R = right;
        for (int i = 1; i < N; i++) {
            L = 0;
            R = right;
            while (L <= R) {//用于查找大于等于arr[i]最左边的位置
                int mid = (L + R) >> 1;
                if (arr[i] > end[mid]) {
                    L = mid + 1;
                } else {
                    R = mid - 1;
                }
            }
            right = max(right, L); //更新右边界
            maxLen = max(maxLen, L + 1);
            end[L] = arr[i];

        }
        return maxLen;
    }
};

三. 最长递增子序列问题III

1.对应letecode链接:

最长上升子序列(三)_牛客题霸_牛客网 (nowcoder.com)

2.题目描述:

最长递增子序列,算法,leetcode

3.解题思路

1.本题只是上题的一个升级版我们只需要定义一个变量记录 一下最长递增子序列的结尾位置在哪里即可,然后再依次遍历

4.对应代码:

class Solution {
  public:
    /**
     * retrun the longest increasing subsequence
     * @param arr int整型vector the array
     * @return int整型vector
     */
    vector<int> LIS(vector<int>& arr) {
        // write code here
        if (arr.empty()) {
            return {};
        }
        int N = arr.size();
        vector<int>dp(N, 1);
        vector<int>end(N);
        int maxLen = 1;
        int maxIndex = 0;
        end[0] = arr[0];
        int right = 0;
        int L = 0;
        int R = right;
        for (int i = 1; i < N; i++) {
            L = 0;
            R = right;
            while (L <= R) {
                int mid = (L + R) >> 1;
                if (arr[i] > end[mid]) {
                    L = mid + 1;
                } else {
                    R = mid - 1;
                }
            }
            right = max(right, L);
            maxLen = max(maxLen, L + 1);
            end[L] = arr[i];
            dp[i] = L + 1;
            if (dp[i] >= maxLen) {//由于要求子典序最小
                maxIndex = i;
                maxLen = dp[i];
            }
        }
        vector<int>ans(maxLen);//获取最长的递增子序列
        for (int i = maxIndex; i >= 0; i--) {
            if (dp[i] == maxLen) {
                ans[--maxLen] = arr[i];
            }
        }
        return ans;
    }
};

思考题:如果要获取所有递增子序列了?文章来源地址https://www.toymoban.com/news/detail-616478.html

#include<iostream>
#include<vector>
using namespace std;
vector<int> process(vector<int>& arr,vector<int>&dp ,int maxLen, int index) {//获取所有最长递增子序列
	vector<int>ans(maxLen);
	for (int i = index; i >= 0; i--) {
		if (dp[i] == maxLen) {
			ans[--maxLen] = arr[i];
		}
	}
	cout << "进来" << endl;
	return ans;

}
int main() {
	int N;
	cin >> N;
	vector<int>arr(N);
	for (int i = 0; i < N; i++) {
		cin >> arr[i];
	}
	vector<int>dp(N, 1);
	vector<int>end(N);
	int maxLen = 1;
	int maxIndex = 0;
	end[0] = arr[0];
	int right = 0;
	int L = 0;
	int R = right;
	for (int i = 1; i < N; i++) {
		L = 0;
		R = right;
		while (L <= R) {
			int mid = (L + R) >> 1;
			if (arr[i] > end[mid]) {
				L = mid + 1;
			}
			else {
				R = mid - 1;
			}
		}
		right = max(right, L);
		maxLen = max(maxLen, L + 1);
		end[L] = arr[i];
		dp[i] = L + 1;
		if (dp[i] >= maxLen) {
			maxIndex = i;
			maxLen = dp[i];
		}
	}
	
	vector<vector<int>>ans;
	for (int i = 0; i < N; i++) {
		if (dp[i] == maxLen) {
			ans.push_back(process(arr, dp, maxLen, i));
		}
	}
	for (int i = 0; i < ans.size(); i++) {
		for (int j = 0; j < ans[0].size(); j++) {
			cout << ans[i][j] << " ";
		}
		cout << endl;
	}


	return 0;
}

到了这里,关于最长递增子序列问题(你真的会了吗)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • LeetCode | C++ 动态规划——300.最长递增子序列、674. 最长连续递增序列、718. 最长重复子数组

    300题目链接 dp 数组定义 dp[i] 表示 i 之前包括 i 的以 nums[i]结尾 的最长递增子序列的长度 需要包含nums[i]结尾,不然在做递增比较的时候,就没有意义了。 递推公式 位置 i 的最长递增子序列 等于 j 从 0 到 i - 1各个位置的最长递增子序列 + 1 的 最大值 if (nums[i] nums[j]) dp[i] = ma

    2024年02月16日
    浏览(48)
  • leetcode300. 最长递增子序列 子序列(不连续)

    https://leetcode.cn/problems/longest-increasing-subsequence/ 给你一个整数数组 nums ,找到其中最长严格递增子序列的长度。 子序列 是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。例如,[3,6,2,7] 是数组 [0,3,1,6,2,2,7] 的子序列。 LIS即最长上升子序列,指

    2024年02月14日
    浏览(43)
  • 算法刷题Day 52 最长递增子序列+最长连续递增子序列+最长重复子数组

    我自己想出来的方法,时间复杂度应该是 O(n2) 滑动窗口 连续的话,可以考虑用滑动窗口 动态规划 贪心算法

    2024年02月14日
    浏览(54)
  • leetcode300. 最长递增子序列(动态规划-java)

    来源:力扣(LeetCode) 链接:https://leetcode.cn/problems/longest-increasing-subsequence 给你一个整数数组 nums ,找到其中最长严格递增子序列的长度。 子序列 是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。例如,[3,6,2,7] 是数组 [0,3,1,6,2,2,7] 的子序

    2024年02月15日
    浏览(44)
  • 【LeetCode: 673. 最长递增子序列的个数 | 动态规划】

    🚀 算法题 🚀 🌲 算法刷题专栏 | 面试必备算法 | 面试高频算法 🍀 🌲 越难的东西,越要努力坚持,因为它具有很高的价值,算法就是这样✨ 🌲 作者简介:硕风和炜,CSDN-Java领域新星创作者🏆,保研|国家奖学金|高中学习JAVA|大学完善JAVA开发技术栈|面试刷题|面经八股文

    2024年02月03日
    浏览(64)
  • ( 动态规划) 674. 最长连续递增序列 / 718. 最长重复子数组——【Leetcode每日一题】

    难度:简单 给定一个未经排序的整数数组,找到最长且 连续递增的子序列 ,并返回该序列的长度。 连续递增的子序列 可以由两个下标 l 和 r(l r) 确定,如果对于每个 l = i r ,都有 nums[i] nums[i + 1] ,那么子序列 [nums[l], nums[l + 1], ..., nums[r - 1], nums[r]] 就是连续递增子序列。

    2024年02月05日
    浏览(52)
  • 动态规划算法 | 最长递增子序列

    通过查阅相关资料 发现动态规划问题一般就是求解最值问题 。这种方法在解决一些问题时应用比较多,比如求最长递增子序列等。 有部分人认为动态规划的核心就是:穷举。因为要求最值,肯定要把所有可行的答案穷举出来,然后在其中找最值。 首先,笔者认为动态规划中

    2024年02月06日
    浏览(54)
  • 贪心算法学习——最长单调递增子序列

    目录 ​编辑 一,题目 二,题目接口 三,解题思路和代码 给你一个整数数组  nums  ,找到其中最长严格递增子序列的长度。 子序列  是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。例如, [3,6,2,7]  是数组  [0,3,1,6,2,2,7]  的子序列。  

    2024年02月08日
    浏览(42)
  • 【动态规划】求最长递增子序列问题

    最长递增子序列(Longest Increasing Subsequence, LIS ) 子序列:对于任意序列s,它的子序列是通过删除其中零个或多个元素得到的另⼀个序列 注:剩余元素的相对顺序保持不变 给定n个整数组成的序列 s [ 1... n ] s[1...n] s [ 1... n ] ,求最长递增子序列LIS(的长度) 8 3 6 1 3 5 4 7 假设

    2024年02月03日
    浏览(50)
  • C++二分查找算法的应用:最长递增子序列

    C++二分算法应用:最长递增子序列 二分查找算法合集 单调映射 点击下载源码 给你一个整数数组 nums ,找到其中最长严格递增子序列的长度。 子序列 是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。例如,[3,6,2,7] 是数组 [0,3,1,6,2,2,7] 的子

    2024年02月06日
    浏览(44)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包