11.最长上升子序列(LIS)

article/2025/9/20 17:32:34

视频讲解:最长上升子序列_哔哩哔哩_bilibili 


解题思路:

1.最长上升子序列的含义是从给定的数中选取尽量多的数字形成单调递增的序列

2.可以把以每一个数字形成单调结尾的方案数看待成一个子问题,然后对后面的子问题提供最优解

3.设定状态,dp【x】为以第x个位置上的数字结尾的序列长度

4.状态转移,如果以第x个位置上的数字结尾,前面a[j]<a[x]的,说明可以结尾,所以就是dp【x】=max(dp【x-1】+1);这里为什么是最大值呢?如果不行的话,则dp【x】=1;

5.初始化dp数组为1,因为本身就是一个单调递增的序列


#include<bits/stdc++.h>
using namespace std;
int a[10010],dp[10010];
int main()
{int n;cin>>n;for(int i=1;i<=n;i++)//输入元素到数组中 cin>>a[i];for(int i=1;i<=n;i++)//初始化dp数组,最小值都为1 dp[i]=1;for(int i=2;i<=n;i++)//从第二项开始转移 {for(int j=1;j<=i;j++)//从他前面开始找 {if(a[i]>a[j])//如果可以a[i]结尾 {dp[i]=max(dp[j]+1,dp[i]);//状态转移方程 }}}for(int i=1;i<=n;i++)//输出方案数 cout<<dp[i]<<" ";return 0;
}

算法优化:

1.上面的思路本质上是在暴力枚举,把每个状态都比较一次,所以时间复杂度是O(n*n),在其中,有很多是没有必要去枚举的,导致浪费了更多的时间,当数据量较大的时候便超时

2.可以利用贪心的思想,如果设置f【i】为此时子序列长度为i的时候末尾的元素值,那么这个元素值较小的时候,才能为后面空出更多的空间

3.设置初始状态f【1】=a【1】,即第一个元素本身就是一个数位,从第二项开始,判断如果a[i]>f【len】的,说明子序列的长度又可以加1,先把a【i】放在新序列的末尾,长度len++,

 即f【++len】=a【i】;

4.当a【i】是小于f【len】的时候,说明他其实可以作为前面子序列中的末尾值,举个例子:

       原序列为:1 3 2 4 5,1作为上升子序列的第1项即f【1】=1;

                         3作为上升子序列的第二项f【2】=3

                         当到达2的时候,发现2比3小,那么此时的2完全可以取代3,因为2可以为后面腾                             出更多的空 间,所以f【2】=2

                        (因为这里只要求我们输出最长上升子序列的长度,所以不必考虑第二个位                                      置 到底是谁,只要确定目前有2个上升的序列长度即可!)

5.那么如何去查找呢?将此时更小的a【i】放到最合适的位置上,这里采用二分查找的方法(也就是真正降低时间复杂度的关键O(nlogn)),因为形成的f【len】已经值单调递增的序列,所以可以利用在排好序的序列中,将a【i】一直往前找,

举个例子:此时1,4,5已经是排好序的子序列,当前最长上升长度为3,也就是5的下标,遍历2的时候,发现比前面的值小,贪心思想来说,2更应该作为末尾值,所以往前找,直到找到比它刚刚小的值,然后放入到该位置后面即可

元素值14523
下标12345

6.最后输出f数组的长度即为最长上升子序列的长度

#include<bits/stdc++.h>
using namespace std;
const int MAXN=99999999;
int a[100010],f[100010];int main()
{int n;cin>>n;for(int i=1;i<=n;i++){cin>>a[i];f[i]=MAXN; //设置最大值,方便后续替换 }f[1]=a[1];//以第一个元素作为子序列的第一个值 int len=1;//记录当前上升子序列的长度 for(int i=2;i<=n;i++){int low=0,high=len,mid;//设置二分查找的区间 if(a[i]>f[len])//如果当前值是大于排好序的末尾元素值的 f[++len]=a[i];//加长子序列的长度,将该值作为该位置的末尾元素值else{while(low<high)//二分查找 当low>=high停止循环 {mid=(high+low)/2;if(f[mid]>a[i])//如果中间值大于a[i],继续往前找 high=mid;//上限更新 else//否则的话 low=mid+1;//下限更新 }f[low]=min(f[low],a[i]);//更新末尾数字 } }cout<<len;return 0;
}


http://chatgpt.dhexx.cn/article/t7fCLSac.shtml

相关文章

C++动态规划之最长上升子序列

1 子序列与上升子序列 1.1 子序列 一个序列A{a1,a2,...an}中任意删除若干项&#xff0c;剩余的序列叫做A的一个子序列。例如序列A{1,3,5,4,2}&#xff0c;删除其中的第3项和第5项&#xff0c;得到序列B{1,3,4}&#xff0c;删除其中的第3项和第4项&#xff0c;得到序列C{1&#…

最长上升子序列(c++图文详解)

这题思路是这样&#xff0c;假设这个序列长度为n&#xff0c;存在数组a中&#xff0c;maxlen[i]表示以第i个数为终点的最长上升子序列的长度&#xff0c;它被初始化为1&#xff0c;因为一开始单个字符的最长上升子序列都是1&#xff08;它自己&#xff09;&#xff0c;我们先用…

最长上升子序列(动态规划)

子序列 所谓的子序列就是在原来序列中找出一部分组成的序列。 与子段不同&#xff0c;不需要连续的某一段&#xff0c;但是要保持原序列的先后顺序 最长上升子序列 在子序列的基础上&#xff0c;后一项大于前一项。 【题目描述】 【输入格式】 【输出格式】 【输入样例】 1…

最长上升子序列 (LIS) 详解+例题模板 (全)

欢迎访问https://blog.csdn.net/lxt_Lucia&#xff5e;&#xff5e; 宇宙第一小仙女\(^o^)/&#xff5e;萌量爆表求带飞≡Σ((( つ^o^)つ~ dalao们点个关注呗&#xff5e; --------------------------------我只是一条可爱哒分界线------------------------------- 1.摘要&…

Import Dependency Management with Exclusion

文章来源: Import Dependency Management with Exclusion Exclusion at import won’t work, try excluding it from the actual user of the dependency 意思是&#xff1a; 在dependencyManagement里面的dependency中&#xff0c;使用exclusions&#xff0c;是不会有作用的…

maven配置中的 scope, type,optional, classifier, 传递依赖,exclusions解释

scope用于依赖范围控制,它管理哪些依赖在哪些classpath 中可用&#xff0c;哪些依赖包含在一个应用中. 它的取值列表如下: 关于为什么使用provided 引入 servlet-api,jsp-api的问题的澄清: 在eclispe里创建web项目时&#xff0c;eclipse为我们自动添加了这两个jar包&#xff0…

记录--对于$off,Exclude 和 Extract的一点理解

这里给大家分享我在网上总结出来的一些知识&#xff0c;希望对大家有所帮助 一.typescript 高阶类型 Exclude 和 Extract Exclude<T, U> TypeScript 2.8 中增加了 Exclude 类型&#xff0c;该如何理解这个高级类型的定义呢&#xff1f; type Exclude<T, U> T exte…

[1120]Maven依赖冲突解决之exclusions

1. 背景 1、作为java生态下开发者&#xff0c;往往需要使用大量线程的第三方库&#xff0c;一般都是以jar包形式存在。 2、maven作为事实上主流的jar包依赖管理工具&#xff0c;Idea和Eclipse都支持创建maven工程来管理jar包依赖。 3、使用maven进行jar包依赖管理时&#xff0…

logical exclusive 与 physical exclusive 的区别

数字电路中一般会有多个clock&#xff0c;这些clock 相互之间有些是同步的&#xff0c;需要做 timing check 的&#xff0c;有些是异步的&#xff0c;不需要做 timing check 的&#xff0c;还有些是互斥的&#xff0c;需要 exclude 掉的&#xff0c;这些关系就需要在 sdc 中声明…

Maven中 排除依赖 exclusions

使用maven进行jar包依赖管理时&#xff0c;maven会自行管理jar包及其依赖链条&#xff0c;但往往会遇到依赖冲突问题&#xff0c;这时候就可以尝试使用exclusions来进行依赖管理 demo:排除tomcat 启用 jetty <dependency><groupId>org.springframework.boot</g…

Exclusive or

题目连接 题意&#xff1a; 每次给一个n&#xff0c;求 (2≤n<10500) 分析&#xff1a; 先说一下自己的想法&#xff0c;如果将n换成二进制数&#xff0c;也就一两千位左右&#xff0c;那么一位一位处理是可以接受的。将0-n写成二进制形式后&#xff0c;显然所有数某一个二进…

Maven依赖冲突解决之exclusions

Maven依赖冲突解决之exclusions 1. 背景 作为java生态下开发者,往往需要使用大量线程的第三方库,一般都是以jar包形式存在。maven作为事实上主流的jar包依赖管理工具,Idea和Eclipse都支持创建maven工程来管理jar包依赖。使用maven进行jar包依赖管理时,maven会自行管理jar包…

【带你吃透C++】引用详解(引用和指针的区别)

引用 1 引用概念 引用不是新定义一个变量&#xff0c;而是给已存在变量取了一个别名&#xff0c;编译器不会为引用变量开辟内存空间&#xff0c;它和它引用的变量共用同一块内存空间。 比如&#xff1a;李逵&#xff0c;在家称为"铁牛"&#xff0c;江湖上人称"…

浅析C++中引用与指针的区别

引用和指针的概念及区别 1. 引用及指针概念1.1 指针概念1.2 引用概念 2. 引用与指针的区别 1. 引用及指针概念 如果熟悉指针和引用的使用&#xff0c;应该能感觉到指针和引用在很多场景使用起来还是有很大的相似性的&#xff0c;尽管它们在语法层面上是俩个完全不同的概念。 那…

C语言-引用和指针的区别?

①引用的格式&#xff1a;数据类型 &引用名 变量名&#xff1b; 指针的格式&#xff1a;数据类型 *变量名指向的变量地址&#xff1b;②使用引用一定要进行初始化 指针为了不出现野指针&#xff0c;也要进行初始化为NULL ③引用只能对数组的元素使用&#xff0c;不能对整个…

c++引用与指针的区别

目录 一、从语法上来讲 二、从汇编层面来看 一、从语法上来讲 1.指针是存储某个实例的地址&#xff0c;引用是实例的别名 2.程序为指针分配内存区域&#xff0c;而不为引用分配内存区域 3.指针使用时要加 “ * ”&#xff0c;解引用&#xff0c;引用可以直接使用 例&…

C++引用和指针的区别

作者&#xff1a;RainMan 链接&#xff1a;https://www.zhihu.com/question/37608201/answer/545635054 来源&#xff1a;知乎 著作权归作者所有。商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处。 引用是C引入的重要机制&#xff08;C语言没有引用&#xff0…

引用和指针的区别

引用和指针的区别: C的引用(Reference) 1.定义引用就是给某个变量起别名&#xff0c;对引用的操作和对该变量操作完全相同。 int a 10&#xff1b; int& b a;//b就是a的别名 b; cout << a << endl;//11 2 常引用 1&#xff09;定义引用时加const修饰&#…

C++中引用和指针的区别

下面用通俗易懂的话来概述一下&#xff1a; 指针-对于一个类型T&#xff0c;T*就是指向T的指针类型&#xff0c;也即一个T*类型的变量能够保存一个T对象的地址&#xff0c;而类型T是可以加一些限定词的&#xff0c;如const、volatile等等。见下图&#xff0c;所示指针的含义&am…

C/C++引用和指针的区别

为什么C/C语言使用指针&#xff1f; 答案&#xff1a;①一方面&#xff0c;每一种编程语言都使用指针。不止C/C使用指针。 每一种编程语言都使用指针。C将指针暴露给了用户(程序员)&#xff0c;而Java和C#等语言则将指针隐藏起来了。 “Everything uses pointers. C just expo…