【DP】最长上升子序列

一.定义

最长上升子序列(Longest Increasing Subsequence,简称LIS)是一个在数列中寻找一个子序列,使得这个子序列中的元素按照升序/降序排列,并且长度最长


二.例题引入

P1020 [NOIP1999 普及组] 导弹拦截 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

我们暂且只考虑第一问


三.分析


四.解法一(O(n^2))

即两重for循环,暴力即可

cpp 复制代码
#include<bits/stdc++.h>
#define maxn 10005
using namespace std;
int a[maxn];
int dp[maxn];
int n,ans;
int main(){
	while(~scanf("%d",&a[++n])); --n;
	for(int i=1;i<=n;i++){
		dp[i]=1;
		for(int j=1;j<=n;j++){
			if(a[j]>a[i] && dp[j]+1>dp[i]){
				dp[i]=dp[j]+1;
			}
		}
		ans=max(ans,dp[i]);
	}
	cout<<ans;
	return 0;
}

五.解法二(O(n logn))

这里我们用最长上升子序列的思维。我只演示过程,数学证明方法是Dilworth 定理,自行查阅。

cpp 复制代码
#include<bits/stdc++.h>
#define maxn 10005
using namespace std;
int a[maxn];
int dp[maxn];
int n,ans;
int top;
int Find(int x){
	int l=1,r=top;
	int mid;
	int ans;
	while(l<=r){
		mid=(l+r)>>1;
		if(dp[mid]<=x){
			ans=mid;
			r=mid-1;
		}else{
			l=mid+1;
		}
	}
	return ans;
}
int main(){
	while(~scanf("%d",&a[++n])); --n;
	dp[++top]=a[1];
	for(int i=2;i<=n;i++){
		if(a[i]<dp[top]) dp[++top]=a[i];
		else{
			int x=Find(a[i]);
			dp[x]=a[i];
		}
	}
	printf("%d",top);
	return 0;
}
相关推荐
贾斯汀玛尔斯4 小时前
每天学一个算法--LSM-Tree(Log-Structured Merge Tree)
java·算法·lsm-tree
浅念-8 小时前
刷穿LeetCode:BFS 解决 Flood Fill 算法
数据结构·c++·算法·leetcode·职场和发展·bfs·宽度优先
做cv的小昊9 小时前
【TJU】研究生应用统计学课程笔记(8)——第四章 线性模型(4.1 一元线性回归分析)
笔记·线性代数·算法·数学建模·回归·线性回归·概率论
贾斯汀玛尔斯9 小时前
每天学一个算法--倒排索引(Inverted Index)
算法·inverted-index
小e说说9 小时前
打破偏科困境:这些学习软件助孩子重燃学习热情
算法
月昤昽10 小时前
autoCAD二次开发 4.正多边形与collection区分
算法·c#·二次开发·autocad二次开发
休息一下接着来10 小时前
C++ 固定容量环形队列实现
c++·算法
im_AMBER11 小时前
手撕hot100之矩阵!看完这篇就AC~
javascript·数据结构·线性代数·算法·leetcode·矩阵
笨笨饿11 小时前
#79_NOP()嵌入式C语言中内联汇编宏的抽象封装模式研究
linux·c语言·网络·驱动开发·算法·硬件工程·个人开发