前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >动态规划C++实现–最长递增子序列

动态规划C++实现–最长递增子序列

作者头像
全栈程序员站长
发布于 2022-09-06 10:13:55
发布于 2022-09-06 10:13:55
51000
代码可运行
举报
运行总次数:0
代码可运行

大家好,又见面了,我是你们的朋友全栈君。

题目: 给定数组arr, 返回arr的最长递增子序列

举例:arr = [2, 1, 5, 3, 6, 4, 8, 9, 7], 返回的最长递增子序列为 [1, 3, 4, 8, 9]

要求:如果arr长度为N,请实现时间复杂度为O(NlogN)的方法。

目录: 一、 时间复杂度O(N^2)的方法

二、 时间复杂度O(NlogN)的方法

一、 先介绍时间复杂度O(N^2)的方法,具体过程如下:

1. 生成长度为N的数组dp, dp[i]表示在以arr[i]这个数结尾的情况下,arr[0…i]中的最大递增序列长度。

2. 对第一个数arr[0]来说,令dp[0] = 1,接下来,从左到右依次计算出每个数结尾的情况下的最长递增序列长度。

3. 计算dp[i],如果最长递增子序列以arr[i]结尾,那么arr[0,…,i-1]中所有比arr[i]小的数都可以作为倒数第二个数

所以 dp[i] = max{ dp[j] + 1} (0 <=j < i, arr[j] < arr[i]), 如果arr[0,…,i-1]中所有数都不比arr[i]小,令dp[i] = 1。

4.根据求出的dp数组,得到最长递增子序列。遍历dp数组,找到最大值以及位置,并开始逆序还原出决策路径。

代码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 最长递增序列 <动态规划> <复杂度0(N^2)>
#include<bits/stdc++.h>
using namespace std;
vector<int> getdp1(vector<int> &arr);
vector<int> generateLIS(vector<int> &arr, vector<int>&dp);


int main() {
    vector<int> arr;
    int temp;
    while(cin >> temp){
        arr.push_back(temp);
    }
    vector<int> dp = getdp1(arr);
    vector<int> lis = generateLIS(arr, dp);
    for (int i = 0; i < lis.size(); i++){
        cout << lis[i]<<" ";
    }
    return 0;
}


vector<int> getdp1(vector<int> &arr){
    vector<int> dp(arr.size(), 0);
    for(int i = 0; i < arr.size(); i++){
        dp[i] = 1;
        for (int j = 0; j < i; j++){
            if (arr[j] < arr[i]){
                dp[i] = max(dp[j]+ 1, dp[i]);
            }
        }
    }
    return dp;
}


vector<int> generateLIS(vector<int> &arr, vector<int> &dp){
    int len = 0; int index = 0;
    for (int i = 0; i < dp.size(); i++) { //寻最长递增子序列末尾的位置和值
        if (dp[i] > len) {
            len = dp[i];              // 最长序列长度
            index = i;                // 最长序列末位置
        }
    }
    vector<int> lis(len, 0);
    lis[--len] = arr[index];
    for (int i = index; i >= 0; i--){
        if (arr[i] < arr[index] && dp[i] == dp[index] - 1){  //从后往前找子序列
            lis[--len] = arr[i];
            index = i;
        }
    }
    return lis;
}
/* input
2 1 5 3 6 4 8 9 7
*/
/* output
1 3 4 8 9
*/

编译器:codeblocks

输入:

输出:

二、 再介绍时间复杂度O(NlogN)的方法,具体过程如下:

计算dp数组的过程达到时间复杂度O(NlogN),这里采用二分查找进行优化,先生成一个长度为N的数组ends和变量right.

遍历的过程中ends[0,…,right]有效区,ends[right+1,…,N-1]无效区,

ends[b] = c 表示遍历到目前为止,在所有长度为b+1的递增序列中,最小的结尾数为c.

以arr=[2,1,5,3,6,4,8,9,7]为例进行说明:

1. 初始时,dp[0]=1, ends[0]=2, rights = 0, 有效区 ends[0…0],ends[0] = 2, 长度为1,结尾为2

2. arr[1] = 1, 在有效区ends[0,…0]找最左边大于或等于arr[1]的数,发现ends[0] =2 >arr[1], 表示以arr[1]结尾的最长序列只有

一 个,dp[1] = 1, ends[0] = 1 (用1替换了原来的2)

3. arr[2] = 5, 在有效区ends[0,..0]找最左边大于或等于arr[2]的数,发现并没有,则ends的有效长度+1, end[1] = 5, 有效区

扩大,dp[2] = 2. arr[0,1] = {1, 5}

依此类推:

代码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 最长递增序列 <动态规划> <复杂度0(NlogN)>
#include<bits/stdc++.h>
using namespace std;
vector<int> getdp2(vector<int> &arr);
vector<int> generateLIS(vector<int> &arr, vector<int>&dp);


int main() {
    vector<int> arr;
    int temp;
    while(cin >> temp){
        arr.push_back(temp);
    }
    vector<int> dp = getdp2(arr);
    vector<int> lis = generateLIS(arr, dp);
    for (int i = 0; i < lis.size(); i++){
        cout << lis[i]<<" ";
    }
    return 0;
}


vector<int> getdp2(vector<int> &arr){
    vector<int> dp(arr.size(), 0);
    vector<int> ends(arr.size(), 0);
    ends[0] = arr[0]; dp[0] = 1;
    int right = 0; int l = 0; int r = 0; int m = 0;
    for (int i = 1; i < arr.size(); i++) {
        l = 0;
        r = right;
        while(l <= r) {      //二分法
            m = (l + r) / 2;
            if (arr[i] > ends[m]){
                l = m + 1;
            }else {
                r = m - 1;
            }
        }
        right = max(right, l);
        ends[l] = arr[i];
        dp[i] = l + 1;
    }
    return dp;
}


vector<int> generateLIS(vector<int> &arr, vector<int> &dp){
    int len = 0; int index = 0;
    for (int i = 0; i < dp.size(); i++) { //寻最长递增子序列末尾的位置和值
        if (dp[i] > len) {
            len = dp[i];
            index = i;
        }
    }
    vector<int> lis(len, 0);
    lis[--len] = arr[index];
    for (int i = index; i >= 0; i--){
        if (arr[i] < arr[index] && dp[i] == dp[index] - 1){  //从后往前找子序列
            lis[--len] = arr[i];
            index = i;
        }
    }
    return lis;
}
/* input
2 1 5 3 6 4 8 9 7
*/
/* output
1 3 4 8 9
*/

编译器:codeblocks

输入:

输出:

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/134209.html原文链接:https://javaforall.cn

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022年6月7,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
动态规划之最长递增子序列
则称这个子序列是一个递增子序列。A的所有子序列中,最长的那个就是最长递增子序列(LIS)
灯珑LoGin
2022/10/31
4070
动态规划系列之最长递增子序列问题解答
今天和大家分享的是动态规划经典问题:最长递增子序列问题解答。(似乎BAT等各大公司都喜欢在面试的时候对动态规划系列经典问题进行笔试。 题目描述: 给定一个整数序列: 求其最长递增子序列(LIS)。如果
机器学习算法工程师
2018/03/06
1.2K0
动态规划系列之最长递增子序列问题解答
最长递增子序列(LIS)[通俗易懂]
处理第二个元素 2 的时候判断是否比前面的元素 4 大,没有的话那么以 2 为结尾的 LIS 就是 2,
全栈程序员站长
2022/08/10
1K0
最长递增子序列(LIS)[通俗易懂]
从最长递增子序列学会如何推状态转移方程
也许有读者看了前文 动态规划套路详解,学会了动态规划的套路:找到了问题的「状态」,明确了dp数组/函数的含义,定义了 base case;但是不知道如何确定「选择」,也就是不到状态转移的关系,依然写不出动态规划解法,怎么办?
labuladong
2021/09/23
8970
最长递增子序列LIS的O(nlogn)的求法
最长递增子序列(Longest Increasing Subsequence)是指n个数的序列的最长单调递增子序列。比如,A = [1,3,6,7,9,4,10,5,6]的LIS是1 3 6 7 9 10。我们现在希望编程求出一个给定的数组,我们能得到LIS的长度。 关于LIS的求法使用DP算法的文章也很多,时间复杂度是O(n2),这里,我们介绍一个只需要不到15行的Python代码或者Java代码来实现一个复杂度O(nlogn)的算法。
全栈程序员站长
2022/08/14
6160
最长递增子序列LIS的O(nlogn)的求法
动态规划+二分查找解决最长递增子序列
很多读者反应,就算看了前文 动态规划详解,了解了动态规划的套路,也不会写状态转移方程,没有思路,怎么办?本文就借助「最长递增子序列」来讲一种设计动态规划的通用技巧:数学归纳思想。
帅地
2019/07/23
3.1K1
动态规划+二分查找解决最长递增子序列
动态规划应用--最长递增子序列 LeetCode 300
有一个数字序列包含n个不同的数字,如何求出这个序列中的最长递增子序列长度?比如2,9,3,6,5,1,7这样一组数字序列,它的最长递增子序列就是2,3,5,7,所以最长递增子序列的长度是4。 https://leetcode-cn.com/problems/longest-increasing-subsequence/
Michael阿明
2021/02/20
4030
动态规划应用--最长递增子序列 LeetCode 300
动态规划:最长连续递增序列
题目链接:https://leetcode-cn.com/problems/longest-continuous-increasing-subsequence/
代码随想录
2021/03/16
2K0
动态规划:最长连续递增序列
动态规划经典题之最长上升子序列
最长上升子序列(Longest Increasing Subsequence,LIS),在计算机科学上是指一个序列中最长的单调递增的子序列。
程序员小熊
2021/05/28
9780
动态规划经典题之最长上升子序列
ACM 省赛E题 最长的递增子序列(动态规划+最长递增子序列)--------C语言—菜鸟级
最长的递增子序列 Bobo学会了如何计算ICPCCamp中O(nlogn)中的最长增加子序列(LIS)。 对于那些没有加入ICPCCamp的人来说,召回LIS(a1,a2,…,an)被定义为f [
Fivecc
2022/11/21
4420
动态规划:最长递增子序列
题目链接:https://leetcode-cn.com/problems/longest-increasing-subsequence/
代码随想录
2021/03/16
8730
动态规划:最长递增子序列
【算法专题】动态规划之子序列问题
子序列 是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。 例如,[3, 6, 2, 7] 是数组[0, 3, 1, 6, 2, 2, 7] 的子序列。
YoungMLet
2024/03/01
3000
最长连续递增子序列问题
给定一个长度为N的数组,给定一个长度为N的数组,找出一个最长的单调自增子序列(不一定连续,但是顺序不能乱)。例如:给定一个长度为6的数组A{5, 6, 7, 1, 2,8},则其最长的单调递增子序列为{5,6,7,8},长度为4。
xujjj
2020/05/18
9440
从动态规划到贪心算法:最长递增子序列问题的方法全解析
最长递增子序列(Longest Increasing subsequence,LIS)是一个经典的问题。最长递增子序列是指在一个序列中,以不下降的顺序连续排列的一系列元素的子序列。这个子序列的长度就是最长递增子序列的长度。
DevKevin
2024/03/19
3300
从动态规划到贪心算法:最长递增子序列问题的方法全解析
动态规划经典题之最长上升子序列最终版
在上一篇文章动态规划经典题之最长上升子序列中,采用动态规划的策略将时间复杂度(暴力法)由 O(n * 2^n) 降到 O(n^2),有没有方法能将时间复杂度进一步降为 O(n * lgn)呢?答案是有的,本文采用 “贪心 + 二分查找” 的策略,将时间复杂度进一步降到 O(n * lgn)。
程序员小熊
2021/05/28
9900
动态规划经典题之最长上升子序列最终版
最长递增子序列
最长递增序列不要求数组元素连续问题,返回递增序列长度和递增序列。o(n^2)做法,顺序比较以第i个元素开头的递增序列即可。 利用动态规划来做,假设数组为1, -1, 2, -3, 4, -5, 6, -7。我们定义LIS[N]数组,其中LIS[i]用来表示以array[i]为最后一个元素的最长递增子序列。 使用i来表示当前遍历的位置: 当i = 0 时,显然,最长的递增序列为(1),则序列长度为1。则LIS[0] = 1 当i = 1 时,由于-1 < 1,因此,必须丢弃第一个值,然后重新建立序列。当前的递
机器学习算法工程师
2018/03/06
1.3K0
最喜欢dp动态规划的一次(暑期刷题)
所有的问题可能不止一种方法,但是由于是dp专题,只会讲述dp解题的方法。如果需要别的算法可以看看后续的更新。 同时,这里的dp算法并不一定是最简单的效率最高的解题方法,可能别的算法更适合更方便。
薛定谔方程难
2024/07/25
950
C++——最长递增子序列问题【组合问题中的动态规划】
#include <iostream> //动态规划法:最长递增子序列之和 int IncreaseOrder(int a[],int n); using namespace std; int main() { int n; cout<<"请输入数组长度:"; cin>>n; int a[n]; int i; cout<<"请输入数组元素:"<<endl; for(i=0; i<n; i++) cin>>a[i]; for(i
瑞新
2020/07/07
3940
最长递增子序列python_求最长递增子序列并输出序列
设L=<a1,a2,…,an>是n个不同的实数的序列,L的递增子序列是这样一个子序列Lin=<aK1,ak2,…,akm>,其中k1<k2<…<km且aK1<ak2<…<akm。求最大的m值。
全栈程序员站长
2022/11/17
7350
最长递增子序列python_求最长递增子序列并输出序列
【LeetCode热题100】【动态规划】最长递增子序列
让dp[i]是以nums[i]为结尾的子序列的最长递增长度,遍历nums[i]之前的元素,如果有比nums[i]小的,说明递增子序列可以延申
叶茂林
2024/04/20
1040
推荐阅读
相关推荐
动态规划之最长递增子序列
更多 >
领券
社区富文本编辑器全新改版!诚邀体验~
全新交互,全新视觉,新增快捷键、悬浮工具栏、高亮块等功能并同时优化现有功能,全面提升创作效率和体验
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文