【狂热算法篇】解锁数据潜能:探秘前沿 LIS 算法

news2025/1/6 6:35:44

 嘿,各位编程爱好者们!今天带来的 LIS 算法简直太赞啦 无论你是刚入门的小白,还是经验丰富的大神,都能从这里找到算法的奇妙之处哦!这里不仅有清晰易懂的 C++ 代码实现,还有超详细的算法讲解,让你轻松掌握 LIS 算法的三种解法:暴力、动态规划、贪心加二分查找,彻底搞懂它在数据处理、资源分配和文本处理中的强大应用呢 看完保证让你收获满满,记得点赞,收藏三连,关注我,后续还有更多精彩的算法干货分享哦 让我们一起开启算法的奇妙之旅,提升编程技能,征服代码世界吧!

欢迎拜访:羑悻的小杀马特.-CSDN博客

本篇主题:剖析所谓的LIS算法(C++版)

制作日期:2025.01.03

隶属专栏:美妙的算法世界

目录

​编辑 

本篇简介:

一·LIS算法: 

1.1算法定义及目的:

1.2实现方法:

1.2.1简单的暴力解法:

1.2.2动态规划解法:

1.2.3贪心算法 + 二分查找解法:

2·算法应用场景:

2.1数据分析与挖掘:

2.2资源分配优化:

2.3文本处理:

二·LIS算法例题:

三·本篇小结:


本篇简介:

 本篇以LIS算法展开细致介绍,不同方法的实现如动态规划,贪心二分实现等;并配合实际例题进行应用;望对读者学习LIS算法有帮助。

一·LIS算法: 

1.1算法定义及目的:

LIS(Longest Increasing Subsequence)算法,即最长递增子序列算法。

它的目标是在一个给定的序列(可以是数字序列、字符序列等)中,找到一个子序列,这个子序列中的元素是按照递增顺序排列的,并且在所有符合递增条件的子序列中,长度是最长的。

例如,在序列 [1, 3, 2, 4, 5] 中,最长递增子序列是 [1, 2, 4, 5],长度为 4。这个算法在很多领域都有应用,比如数据挖掘中分析数据的趋势,或者在文本处理中分析文本的某种递增模式。

1.2实现方法:

1.2.1简单的暴力解法:

又称枚举法;下面说一下它的思路:

这种方法是最直观的。它会枚举所有可能的子序列,然后检查每个子序列是否是递增的,并记录下最长的递增子序列。

具体来说,对于一个长度为 n 的序列,它有 2^n 个子序列,需要逐个检查这些子序列是否满足递增条件。

 但是它有个致命的缺点:

时间复杂度非常高,达到了指数级别 O (2^n)。在实际应用中,当序列长度稍微变大时,计算量会变得极其庞大,效率极低。例如,当 n = 20 时,2^20 约等于 100 万次运算,这会导致程序运行时间过长。

代码实现:

// 暴力解法
int lis_brute_force(const std::vector<int>& nums) {
    int n = nums.size();
    int max_length = 0;
    // 枚举所有可能的子序列
    for (int mask = 0; mask < (1 << n); ++mask) {
        std::vector<int> subseq;
        for (int i = 0; i < n; ++i) {
            if (mask & (1 << i)) {
                subseq.push_back(nums[i]);
            }
        }
        bool is_increasing = true;
        for (size_t i = 1; i < subseq.size(); ++i) {
            if (subseq[i] <= subseq[i - 1]) {
                is_increasing = false;
                break;
            }
        }
        if (is_increasing) {
            max_length = std::max(max_length, static_cast<int>(subseq.size()));
        }
    }
    return max_length;
}

代码解释:

使用位运算枚举所有可能的子序列。mask 从 0 到 (1 << n) - 1,对于每个 mask,将对应位为 1 的元素添加到 subseq 向量中。

检查 subseq 是否是递增序列,如果是,更新 max_length

时间复杂度为 o(2^n),效率低,适合较短序列。

因此我们极大不推荐这种解法。

1.2.2动态规划解法:

原理:

1.2.2.1从左往右填表:

动态规划的核心思想是将一个复杂的问题分解为一系列相互关联的子问题,并通过记录子问题的解来避免重复计算。

对于 LIS 算法,我们定义一个状态数组 dp,其中 dp [i] 表示以第 i 个元素结尾的最长递增子序列的长度。

 状态转移方程

dp [i]=max (dp [j]) + 1,其中 j < i 且 nums [j]<nums [i]。

这意味着要找到在 i 之前的元素 j,使得 nums [j] 小于 nums [i],并且 dp [j] 是最大的,然后将 dp [i] 更新为 dp [j]+1。

时间复杂度分析:

 时间复杂度为 O (n^2),相比于暴力解法有了很大的提升。

例如,对于长度为 100 的序列,暴力解法可能需要计算 2^100 次左右,而动态规划只需要计算 100^2 = 10000 次左右,大大减少了计算量。

代码实现:

首先,定义完了dp状态,先明确我们的任务:即从这这段数组中(其实是参差不齐),但是我们忽略降序的数,只从升序的那条路看起,找最长的。

比如我们到了i位置,想要得到的是? 以i位置为结尾的最长(包括i位置);因此我们需要在i之前找一条升序最长路L(末尾元素一定要比i小)--->这里正好就是我们的dp[j](j是由0~i-1);因此我们只需要遍历得到以j为结尾的dp的最大值就好;如果是结尾元素值比我们标定i对应值大;那么这个j就不能作为L的结尾元素;直接else跳过。

故,上面所说的我们最后是遍历得到L然后再+1得到dp[i];因此我们让它更加贴近一下dp状态方程的写法-->下面我们只用维护L+1(dp[i])的最大值即可了

int lis_dp(const std::vector<int>& nums) {
    int n = nums.size();
    std::vector<int> dp(n, 1);
    int max_length = 1;
    for (int i = 1; i < n; ++i) {
        for (int j = 0; j < i; ++j) {
            if (nums[i] > nums[j]) {
                // 状态转移方程 dp[i] = max(dp[i], dp[j] + 1)
                dp[i] = std::max(dp[i], dp[j] + 1);
            }
        }
        max_length = std::max(max_length, dp[i]);
    }
    return max_length;
}

代码解释:

使用 dp 数组存储以每个元素结尾的最长递增子序列的长度。

对于每个元素 i,遍历 i 之前的元素 j,若 nums[i] > nums[j],根据状态转移方程 dp[i] = std::max(dp[i], dp[j] + 1) 更新 dp[i]

最终 max_length 存储了最长递增子序列的长度,时间复杂度为 。

1.2.2.2从右向左填表: 

这里还可以是dp[i]表示以第i个元素为开始的 最长递增子序列的长度;这里只不过就是和上面的填表顺序颠倒一下。

代码展示:

这里的实现思路其实大差不多;我们只是定义的dp状态不同;只是所谓的推导从前变成了后了;

i为开始元素:因此我们要找的是i的右侧的大于i位置值的升序最长路L;因此从i+1开始看(必须要包含i+1为起点的最大路L)【->故我们是当填充i从i右侧找;因此这个是逆向填表(从右往左填表)】;注:遍历时,这里我们符合的要求(L)首先是j位置的值一定要大于i位置;其次就是找最长;因此遍历到长的L;因此我们为了让它像dp状态方程写法,故每次是求L+1作为dp[i]暂定值,然后求个max。

int LIS(vector<int> d) {
	int maxn = 1;
	vector<int> dp(d.size(),1);
	for (int i = d.size() - 2; i >= 0; i--) {
		for (int j = i + 1; j < d.size(); j++) {
			if (d[i] < d[j]) dp[i] = max(dp[i],dp[j] + 1);
		}
		maxn = max(dp[i],maxn);
	}
	return maxn;
}

 这两种定义方法都可以。

1.2.3贪心算法 + 二分查找解法:

贪心思想引入:

贪心算法的策略是在每一步都做出当前看起来最优的选择。对于 LIS 算法,我们维护一个辅助数组 tail,它存储了当前找到的最长递增子序列。当扫描到一个新元素时,我们尽量将它插入到 tail 数组中合适的位置,使得 tail 数组仍然保持递增。

 二分查找:

为了高效地将新元素插入到 tail 数组中,我们使用二分查找。每次插入元素时,在 tail 数组中找到第一个大于等于新元素的位置,然后用新元素替换它。这样可以保证 tail 数组始终是递增的,并且长度尽可能长。

时间复杂度

这种方法的时间复杂度可以达到 O (nlogn),这是一种比较高效的解法。例如,对于长度为 1000 的序列,其计算量比动态规划的 O (n^2) 解法又减少了很多,能够更快地得到结果。


代码实现:

int lis_greedy(const std::vector<int>& nums) {
    std::vector<int> tail;
    for (int num : nums) {
        auto it = std::lower_bound(tail.begin(), tail.end(), num);
        if (it == tail.end()) {
            tail.push_back(num);
        } else {
            *it = num;
        }
    }
    return tail.size();
}

代码解释:

 

维护 tail 向量存储当前找到的最长递增子序列。

对于每个元素 num,使用 std::lower_bound 找到 tail 中第一个大于等于 num 的位置。

如果 it 等于 tail.end(),说明 num 比 tail 中所有元素都大,添加到 tail 末尾;否则,将 tail 中 it 位置的元素更新为 num

时间复杂度为 o(nlogn),是最有效的实现方式 

这种也是比较推荐的!!! 

2·算法应用场景:

2.1数据分析与挖掘

在分析数据的趋势变化时,LIS 算法可以帮助找到数据中的上升趋势部分,例如股票价格的上涨阶段、气温的上升周期等。

2.2资源分配优化

在任务调度或者资源分配场景中,如果任务有先后顺序要求或者资源有递增的利用顺序,LIS 算法可以辅助找到最优的分配方案。

2.3文本处理

在文本编辑软件中,用于分析文本段落中句子的长度递增模式,或者单词的某种语义递增模式,辅助进行文本排版或者语义分析。

二·LIS算法例题:

 下面我们就用一道例题,来应用上面所述的LIS算法解答吧:

测试用例:

 输入:6 1 4 2 2 5 6

 输出:4

题目链接: 蓝桥账户中心

首先我们先看数据范围:

这里我们由题意可以看出:

总结下:就是当选了大的数值后就无法在选择比它小的数值;因此我们可以 选择把它排好升序,但是此时要注意符合刚才说的要求即可,这时就联想到了LIS算法了,但是前两种即暴力,动态规划实现是不合适的(大数据范围),因此后面我们就用贪心+二分来实现。

这里也就是要求我们要是升序,并且不能重复 (去重复:1·利用lower_bound的特性找到相同的会覆盖掉,2·当发现找不到返回的是区间最后+1位置的迭代器也是会覆盖,不记录len(如上图的情况))。

因此直接上手代码:

#include <bits/stdc++.h>
using namespace std;
const int N=3e5+5;
 int r[N]={0};
int main()
{  //最长递增子序列问题(贴近):lis
   int n;
   cin>>n;
  vector<int>v(n+1);
  for(int i=1;i<=n;i++) cin>>v[i];
   int len=0;//r答案数组的长度:确保r数组始终是升序的
   //(不是一般的sort类的升序,而是确保了在r数组中出现的升序元素相对位置符合原数组的相对位置)
   for(int i=1;i<=n;i++){
     if(v[i]>r[len]) r[++len]=v[i];//比它大直接放在后面(多可以个0位置,方便比较)
     else {
       //返回大于它的数完成覆盖:
       //找不到也就是r数组存在于当前值第一大的值,放在len后面的位置,之后会覆盖(相当于相等的数据就是无效的)
        // int pos=lower_bound(r+1,r+len+1,v[i])-r;
        // r[pos]=v[i];

        //或者
        *lower_bound(r+1,r+len+1,v[i])=v[i];
      
     }
   }
   cout<<len;
  return 0;
}

这道题也是不n能用o(N^2)复杂度的动态规划局解法;故就用复杂度为o(nlogn)的贪心二分来解决。

最后也是通过了。 

三·本篇小结:

本篇介绍了LIS算法;下面介绍一下具体怎么用来小结一下:

当我们发现要求的是最长地址子序列,就可以选择它;具体有三种情况;但是合适的要么就动归,要么就二分;根据数据范围选择即可了。

如果我们只要这个序列的长度:两种都可。

但是,如果是还要它的数据那么就只能选二分来模拟了。

二分的话,就要保证:在lis数组中出现的升序元素相对位置符合原数组的相对位置。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2270733.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【漫话机器学习系列】033.决策树回归(Decision Tree Regression)

决策树回归&#xff08;Decision Tree Regression&#xff09; 决策树回归是一种基于树状结构进行回归分析的监督学习方法。它将输入空间递归地划分为多个区域&#xff0c;并在每个区域内拟合一个简单的常数值&#xff0c;从而对目标变量进行预测。 决策树回归的原理 树的构建…

Vue3中使用 Vue Flow 流程图方法

效果图&#xff1a; 最近项目开发时有一个流程图的功能&#xff0c;需要做流程节点的展示&#xff0c;就搜到了 Vue Flow 这个插件&#xff0c;这个插件总得来说还可以&#xff0c;简单已使用&#xff0c;下边就总结一下使用的方法&#xff1a; Vue Flow官网&#xff1a;https…

ArcGIS JSAPI 高级教程 - 通过RenderNode实现视频融合效果(不借助三方工具)

ArcGIS JSAPI 高级教程 - 通过RenderNode实现视频融合效果(不借助三方工具) 核心代码完整代码在线示例地球中展示视频可以通过替换纹理的方式实现,但是随着摄像头和无人机的流行,需要视频和场景深度融合,简单的实现方式则不能满足需求。 三维视频融合技术将视频资源与三维…

Appllo学习

补充学习: Apollo管理多环境下的配置和踩坑实践 - 简书 Apollo-阿波罗配置中心超详细教程_apllo-CSDN博客 springboot本地local配置覆盖远程Apollo配置&#xff08;含Apollo配置加载顺序说明&#xff09;_本地覆盖apollo配置-CSDN博客 Apollo 配置中心详细教程 - 简书 (包含…

React18路由和Vue3路由进行对比

本文将深入比较 React 18 和 Vue 3 路由的不同之处&#xff0c;帮助你更好地理解如何在这两个框架中进行路由管理。希望能对于从 Vue 3 迁移到 React 的开发者&#xff0c;理解这些差异&#xff0c;帮助你更高效地切换框架和构建应用。 1. 路由配置 React 18 的路由配置 Rea…

Windows系统下载、部署Node.js与npm环境的方法

本文介绍在Windows电脑中&#xff0c;下载、安装并配置Node.js环境与npm包管理工具的方法。 Node.js是一个基于Chrome V8引擎的JavaScript运行时环境&#xff0c;其允许开发者使用JavaScript编写命令行工具和服务器端脚本。而npm&#xff08;Node Package Manager&#xff09;则…

浏览器选中文字样式

效果 学习 Chrome: 支持 ::selection。Firefox: 支持 :-moz-selection 和 ::selection。Safari: 支持 ::selection。Internet Explorer: 支持 :-ms-selection。Microsoft Edge: 支持 ::-ms-selection 和 ::selection。 代码 <!DOCTYPE html> <html lang"en&qu…

指代消解:自然语言处理中的核心任务与技术进展

目录 前言1. 指代消解的基本概念与分类1.1 回指与共指 2. 指代消解的技术方法2.1 端到端指代消解2.2 高阶推理模型2.3 基于BERT的模型 3. 事件共指消解&#xff1a;跨文档的挑战与进展3.1 联合模型3.2 语义嵌入模型&#xff08;EPASE&#xff09; 4. 应用场景与前景展望4.1 关键…

CDPHudi实战-集成spark

[一]使用Spark-shell 1-配置hudi Jar包 [rootcdp73-1 ~]# for i in $(seq 1 6); do scp /opt/software/hudi-1.0.0/packaging/hudi-spark-bundle/target/hudi-spark3.4-bundle_2.12-1.0.0.jar cdp73-$i:/opt/cloudera/parcels/CDH/lib/spark3/jars/; done hudi-spark3.4-bu…

leetcode:面试题 17.01. 不用加号的加法(python3解法)

难度&#xff1a;简单 设计一个函数把两个数字相加。不得使用 或者其他算术运算符。 示例: 输入: a 1, b 1 输出: 2 提示&#xff1a; a, b 均可能是负数或 0结果不会溢出 32 位整数 题解&#xff1a; class Solution:def add(self, a: int, b: int) -> int:sum_list [a…

设计模式 结构型 适配器模式(Adapter Pattern)与 常见技术框架应用 解析

适配器模式&#xff08;Adapter Pattern&#xff09;是一种结构型设计模式&#xff0c;它允许将一个类的接口转换成客户端所期望的另一个接口&#xff0c;从而使原本因接口不兼容而无法一起工作的类能够协同工作。这种设计模式在软件开发中非常有用&#xff0c;尤其是在需要集成…

二维码文件在线管理系统-收费版

需求背景 如果大家想要在网上管理自己的文件&#xff0c;而且需要生成二维码&#xff0c;下面推荐【草料二维码】&#xff0c;这个系统很好。特别适合那些制造业&#xff0c;实体业的使用手册&#xff0c;你可以生成一个二维码&#xff0c;贴在设备上&#xff0c;然后这个二维码…

MySQL8安装与卸载

1.下载mysql MySQL :: Download MySQL Community Serverhttps://dev.mysql.com/downloads/mysql/ 2.解压mysql安装包 解压到自己定义的目录&#xff0c;这里解压就是安装&#xff0c;解压后的路径不要有空格和中文。 3.配置环境变量 配置环境变量可以方便电脑在任何的路径…

数据挖掘——关联规则挖掘

数据挖掘——关联数据挖掘 关联数据挖掘关联规则关联规则挖掘问题&#xff1a;具体挖掘过程Apriori 产生关联规则 关联数据挖掘 关联分析用于发现隐藏在大型数据集中的令人感兴趣的联系&#xff0c;所发现的模式通常用关联规则或频繁项集的形式表示。 关联规则反映一个事物与…

【74HC192减法24/20/72进制】2022-5-17

缘由用74ls192设计一个72进制的减法计数器&#xff0c;需要有逻辑电路图-硬件开发-CSDN问答

Samsung手机首次主要采用竞对Micron LPDDR5内存

根据韩国媒体《韩国先驱报》&#xff08;The Korea Herald&#xff09;的报道&#xff0c;即将在1月底发布的三星 Galaxy S25 系列智能手机将首次主要使用美光科技&#xff08;Micron Technology&#xff09;提供的移动DRAM&#xff0c;而非三星自家的产品。这一消息对于三星的…

Linux驱动开发学习准备(Linux内核源码添加到工程-Workspace)

Linux内核源码添加到VsCode工程 下载Linux-4.9.88源码&#xff1a; 没有处理同名文件的压缩包&#xff1a; https://pan.baidu.com/s/1yjIBXmxG9pwP0aOhW8VAVQ?pwde9cv 已把同名文件中以大写命名的文件加上_2后缀的压缩包&#xff1a; https://pan.baidu.com/s/1RIRRUllYFn2…

leetcode题目(3)

目录 1.加一 2.二进制求和 3.x的平方根 4.爬楼梯 5.颜色分类 6.二叉树的中序遍历 1.加一 https://leetcode.cn/problems/plus-one/ class Solution { public:vector<int> plusOne(vector<int>& digits) {int n digits.size();for(int i n -1;i>0;-…

vue3+Echarts+ts实现甘特图

项目场景&#xff1a; vue3Echartsts实现甘特图;发布任务 代码实现 封装ganttEcharts.vue <template><!-- Echarts 甘特图 --><div ref"progressChart" class"w100 h100"></div> </template> <script lang"ts&qu…

接受Header使用错Map类型,导致获取到的Header值不全

问题复现 在 Spring 中解析 Header 时&#xff0c;我们在多数场合中是直接按需解析的。例如&#xff0c;我们想使用一个名为 myHeaderName 的 Header&#xff0c;我们会书写代码如下&#xff1a;RequestMapping(path "/hi", method RequestMethod.GET) public Str…