LIS(最长上升子序列)的三种经典求法
求最长上升子序列的三种经典方案:
题型简介:
给定一个长度为 $ N $ 的数列,求它数值单调递增的子序列长度最大为多少。即已知有数列 $ A $ , $ A={A_1,A_2....A_n} $ ,求 $ A $ 的任意子序列 $ B $ ( $ B={A_{k_1},A_{k_2}....A_{k_p}} $ ),使 $ B $ 满足 $ k_1<k_2<....<k_p $ 且 $ A_{k_1}<A_{k_2}<....<A_{k_p} $ 。现求 $ p $ 的最大值。
$ solution\quad 1: $
先说一种最普遍的方法,因为所求为子序列,所以这道题很容易想到一种线性动态规划。我们需要求最长上升子序列,为了上升我们肯定要知道我们当前阶段最后一个元素为多少,为了最长我们还要知道当前我们的序列有多长。我们可以用前者来充当第一维描述:设 $ F[i] $ 表示以 $ A[i] $ 为结尾的最长上身子序列的长度,为了保证保证元素单调递增我们肯定只能从 $ i $ 前面的且末尾元素比 $ A[i] $ 小的状态转移过来:
$ F[i]=^{max}_{0\leq j<i,A[j]<a[i]}\{F[j]+1\} $
初始值为 $ F[0]=0 $ ,而答案可以是任何阶段中只要长度最长的那一个,所以我们边转移边统计答案。
复杂度: $ O(n^2) $
$ code\quad 1: $
#include<iostream>
#include<cstdio>
#define ll long long
#define rg register int
using namespace std;
int n,ans;
int a[10005];
int f[10005];
int main(){ cin>>n;
for(rg i=1;i<=n;++i) cin>>a[i];
for(rg i=1;i<=n;++i){
for(rg j=1;j<i;++j) //枚举转移
if(a[j]<a[i])f[i]=max(f[i],f[j]);
++f[i]; ans=max(ans,f[i]); //更新答案
}cout<<ans<<endl;
return 0;
}
$ solution\quad 2: $
我们发现上一种方法会枚举前面较小的位置,我们考虑能否用数据结构优化,首先将转移方程列一下:
$ F[i]=^{max}_{0\leq j<i,A[j]<a[i]}\{F[j]+1\} $
我们发现大括号中的 $ 1 $ 与 $ j $ 没有任何关系,所以我们将它提取出来:
$ F[i]=1+~~^{max}_{0\leq j<i,A[j]<a[i]}\{F[j]\} $
然后我们发现我们只需要将比 $ i $ 小的所有的符合 $ A[j]<A[i] $ 的 $ F[j] $ 的最大值求出来,但是这个条件 $ A[j]<A[i] $ 实在是太麻烦了,所以我们换一种思维方法:对于原序列每个元素,它有一个下标和一个权值,最长上升子序列实质就是求最多有多少元素它们的下标和权值都单调递增。
于是我们将 $ A $ 数组的每一个元素先记下他现在的下标,然后按照权值从小到大排序。接着我们按从小到大的顺序枚举 $ A $ 数组,(此时权值已经默认单调递增了)我们的转移也就变成从之前的标号比它小的状态转移过来,这个我们只需要建立一个与编号为下标维护长度的最大值的树状数组即可,枚举 $ A $ 数组时按元素的序号找到它之前序号比他小的长度最大的状态更新,然后将它也加入树状数组中。 期望复杂度: $ O(nlog(n)) $
$ code\quad 2: $
#include<iostream>
#include<cstdio>
#include<algorithm>
#define ll long long
#define rg register int
using namespace std;
int n;
int s[200005];
struct su{
int v,id; //按照权值为第一关键字保证算法正确性
inline bool operator <(su x){
if(v==x.v)return id>x.id; //按照序号从大到小可以保证所求为上升子序列
return v<x.v; //(针对标号)因为相同权值的数,前面的状态不能转移给后面
} //(针对标号)从大到小枚举就不会出现这种情况
}a[200005];
inline void add(int x,int y){
for(;x<=n;x+=x&-x) s[x]=max(s[x],y);
}
inline int ask(int x){
rg res=0;
for(;x>=1;x-=x&-x) res=max(s[x],res);
return res;
}
int main(){
cin>>n;
for(rg i=1;i<=n;++i)
cin>>a[i].v,a[i].id=i;
sort(a+1,a+n+1);
for(rg i=1;i<=n;++i)
add(a[i].id,ask(a[i].id)+1);
cout<<ask(n)<<endl;
return 0;
}
关于树状数组求最长上升子序列的方案及方案数,这个需要结构体来实现,构建结构体数组使其中每一个元素可以包含多个信息,这样在树状数组更新时可以做到顺便兼顾记录前驱,以及累计方案数(需要去重)。关于具体如何实现,可以参见我出的这场考试中的第二题:五彩棒
另外真的很抱歉,博主现在拿的平板,家里电脑坏了,不能具体解答。
$ solution\quad 3: $
这是最快的方法:贪心加二分查找
我之前说过:我们肯定要知道我们当前阶段最后一个元素为多少,还有当前我们的序列有多长。前两种方法都是用前者做状态,我们为什么不可以用后做状态呢?:设 $ F[i] $ 表示长度为 $ i $ 的最长上升子序列的末尾元素的最小值,我们发现这个数组的权值一定单调不降(仔细想一想,这就是我们贪心的来由)。于是我们按顺序枚举数组 $ A $ ,每一次对 $ F[] $ 数组二分查找,找到小于 $ A[i] $ 的最大的 $ F[j] $ ,并用它将 $ F[j+1] $ 更新。
注意:这个方法虽快,但是讲实话还是树状数组好一些,因为对于最长上升子序列的方案输出和计算方案数(upd:很抱歉咕掉了,现在补一下坑,在上面第二种方法结尾),树状数组有很多优势!二分查找因为贪心的缘故会被限制。
期望复杂度: $ O(nlogn) $
$ code\quad 3: $
#include<iostream>
#include<cstdio>
#include<algorithm>
#define ll long long
#define rg register int
using namespace std;
int n;
int a[200005];
int f[200005];
int main(){
cin>>n;
for(rg i=1;i<=n;++i) cin>>a[i];
rg ans=1; f[1]=a[1];
for(rg i=2;i<=n;++i){
rg l=1,r=ans,mid;
while(l<=r){
mid=(l+r)>>1;
if(a[i]<=f[mid])r=mid-1;
else l=mid+1;
}f[l]=a[i];
if(l>ans)++ans;
}cout<<ans<<endl;
return 0;
}
LIS(最长上升子序列)的三种经典求法的更多相关文章
- 算法设计 - LCS 最长公共子序列&&最长公共子串 &&LIS 最长递增子序列
出处 http://segmentfault.com/blog/exploring/ 本章讲解:1. LCS(最长公共子序列)O(n^2)的时间复杂度,O(n^2)的空间复杂度:2. 与之类似但不同的 ...
- POJ - 3903 Stock Exchange(LIS最长上升子序列问题)
E - LIS Time Limit:1000MS Memory Limit:65536KB 64bit IO Format:%I64d & %I64u Descripti ...
- hdu 5256 序列变换(LIS最长上升子序列)
Problem Description 我们有一个数列A1,A2...An,你现在要求修改数量最少的元素,使得这个数列严格递增.其中无论是修改前还是修改后,每个元素都必须是整数. 请输出最少需要修改多 ...
- POJ 3903 Stock Exchange (E - LIS 最长上升子序列)
POJ 3903 Stock Exchange (E - LIS 最长上升子序列) 题目链接:http://acm.hust.edu.cn/vjudge/contest/view.action ...
- 动态规划模板1|LIS最长上升子序列
LIS最长上升子序列 dp[i]保存的是当前到下标为止的最长上升子序列的长度. 模板代码: int dp[MAX_N], a[MAX_N], n; int ans = 0; // 保存最大值 for ...
- POJ 1887 Testingthe CATCHER (LIS:最长下降子序列)
POJ 1887Testingthe CATCHER (LIS:最长下降子序列) http://poj.org/problem?id=3903 题意: 给你一个长度为n (n<=200000) ...
- LIS最长上升子序列三种方法 (模板)
O(n^)的方法: #include <iostream> #include <stdio.h> #include <cstring> #include <a ...
- 【noi 2.6_1759】LIS 最长上升子序列(DP,3种解法)
题意我就不写了.解法有3种: 1.O(n^2).2重循环枚举 i 和 j,f[i]表示前 i 位必选 a[i] 的最长上升子序列长度,枚举a[j]为当前 LIS 中的前一个数. 1 #include& ...
- LIS(最长上升子序列)的 DP 与 (贪心+二分) 两种解法
正好训练赛来了一道最长递减序列问题,所以好好研究了一下最长递增序列问题. B - Testing the CATCHER Time Limit:1000MS Memory Limit:3000 ...
随机推荐
- Understanding performance, load and stress testing
What are performance, load and stress testing? Performance testing, load testing and stress testing ...
- 【bzoj4868】[Shoi2017]期末考试 前缀和+暴力
题目描述 有n位同学,每位同学都参加了全部的m门课程的期末考试,都在焦急的等待成绩的公布.第i位同学希望在第ti天或之前得知所.有.课程的成绩.如果在第ti天,有至少一门课程的成绩没有公布,他就会等待 ...
- LightOJ——1012Guilty Prince(连通块并查集)
1012 - Guilty Prince Time Limit: 2 second(s) Memory Limit: 32 MB Once there was a king named Akbar. ...
- Classloader中loadClass()方法和Class.forName()区别
Classloader中loadClass()方法和Class.forName()都能得到一个class对象,那这两者得到的class对象有什么区别呢 1.java类装载的过程 Java类装载有三个步 ...
- 用 Jackson 来处理 JSON
Jackson 是一个 Java 用来处理 JSON 格式数据的类库,性能非常好. 首先创建一个User对象类 (User.java) package com.sivalabs.json; impor ...
- Contest Hunter #46 T1 磁力块 [分块]
描述 在一片广袤无垠的原野上,散落着N块磁石.每个磁石的性质可以用一个五元组(x,y,m,p,r)描述,其中x,y表示其坐标,m是磁石的质量,p是磁力,r是吸引半径.若磁石A与磁石B的距离不大于磁石A ...
- 「CodePlus 2018 3 月赛」白金元首与莫斯科
$n \leq 17,m \leq 17$,$n*m$的01矩形,对每一个0问:当他单独变成1之后,在其他0处放多米诺牌(不一定放满,可以不放)的方案数.膜$1e9+7$. 直接$dp$是$n^42^ ...
- Codeforces Round #275 (Div. 2) B. Friends and Presents 二分+数学
8493833 2014-10-31 08:41:26 njczy2010 B - Friends and Presents G ...
- 数字IC设计入门必备——VIM自定义模板调用与VCS基本仿真操作示例
一.前言 毕业论文答辩结束,闲下来写篇文章.芯片研发人员都在Linux系统下借助各种EDA工具和代码语言完成工作,因此提高代码开发效率,熟练运用开发工具是十分必要的.本文讲述VIM编辑神器的veril ...
- git(二):一些简单入门命令
一.创建仓储(版本库) 可以创建在空目录下创建git仓库,也可以在已有项目里创建git仓储. $ mkdir NewName //仓储名 $ cd Newname //进入到该仓储目录中 $ git ...