动态规划---最长上升子序列问题(O(nlogn),O(n^2))
LIS(Longest Increasing Subsequence)最长上升子序列 或者 最长不下降子序列。很基础的题目,有两种算法,复杂度分别为O(n*logn)和O(n^2) 。
*********************************************************************************
先回顾经典的O(n^2)的动态规划算法:
设a[t]表示序列中的第t个数,dp[t]表示从1到t这一段中以t结尾的最长上升子序列的长度,初始时设dp[t] = 0(t = 1, 2, ..., len(A))。则有动态规划方程:dp[t] = max{1, dp[j] + 1} (j = 1, 2, ..., t - 1, 且a[j] < a[t])。
一般若从a[t]开始,此时最长不下降子序列应该是按下列方法求出的:
在a[t+1],a[t+2],...a[n]中,找出一个比a[t]大的且最长的不下降子序列,作为它的后继。
代码实现如下:
#include<iostream>
using namespace std;
#define max(a,b) a>b?a:b
int main()
{
int n, i, j, dp[], x[], max_len;
while (cin >> n)
{
for (i = ; i < n; i++)
cin >> x[i];
dp[] = ;//表示以x[0]为子序列最右边的长度位1
for (i = ; i < n; i++)
{
dp[i] = ;//初始化每种情况最小值为1
for (j = ; j < i; j++)
{
if (x[i]>x[j] && dp[j] + >dp[i])//从0-i进行扫描,查找边界小于当前最优解长度相等的解优化最优解
dp[i] = dp[j] + ;//如果允许子序列相邻元素相同 x[i]>=x[j]&&dp[j]+1>dp[i];
}
}
for (i = max_len = ; i < n; i++)
max_len = max(max_len, dp[i]);//等到最大子序列长度
cout << max_len << endl;
}
return ;
}
最长上升子序列O(nlogn)解法
在一列数中寻找一些数,这些数满足:任意两个数a[i]和a[j],若i<j,必有a[i]<a[j],这样最长的子序列称为最长递增子序列。
设dp[i]表示以i为结尾的最长递增子序列的长度,则状态转移方程为:
dp[i] = max{dp[j]+1}, 1<=j<i,a[j]<a[i].
考虑两个数a[x]和a[y],x<y且a[x]<a[y],且dp[x]=dp[y],当a[t]要选择时,到底取哪一个构成最优的呢?显然选取a[x]更有潜力,因为可能存在a[x]<a[z]<a[y],这样a[t]可以获得更优的值。在这里给我们一个启示,当dp[t]一样时,尽量选择更小的a[x].
按dp[t]=k来分类,只需保留dp[t]=k的所有a[t]中的最小值,设g[k]记录这个值,g[k]=min{a[t],dp[t]=k}。
这时注意到g的两个特点(重点):
1. g[k]在计算过程中单调不升;
2. g数组是有序的,g[1]<g[2]<..g[n]。
利用这两个性质,可以很方便的求解:
(1).设当前已求出的最长上升子序列的长度为len(初始时为1),每次读入一个新元素x:
(2).若x>g[len],则直接加入到d的末尾,且len++;(利用性质2)
否则,在g中二分查找,找到第一个比x小的数g[k],并g[k+1]=x,在这里x<=g[k+1]一定成立(性质1,2)。
代码实现如下:
#include<iostream>
#include<cstring>
using namespace std;
const int maxn = ;
int binary_search(int key, int *g, int low, int high)
{
while (low < high)
{
int mid = (low + high) >> ;
if (key >= g[mid])
low = mid + ;
else
high = mid;
}
return low;
}
int main()
{
int i, j, a[maxn], g[maxn], n, len;
while (cin >> n)
{
memset(g, , sizeof(g));
for (i = ; i < n; i++)
cin >> a[i];
g[] = a[], len = ;//初始化子序列长度为1,最小右边界
for (i = ; i < n; i++)
{
if (g[len] < a[i])//(如果允许子序列相邻元素相同 g[len]<=a[i],默认为不等)
j = ++len; //a[i]>g[len],直接加入到g的末尾,且len++
else
j = binary_search(a[i], g, , len + );
g[j] = a[i];//二分查找,找到第一个比a[i]小的数g[k],并g[k+1]=a[i]
}
cout << len << endl;
}
return ;
}
例题分析:(swust oj 126 低价购买)
低价购买
这里是某支股票的价格清单:
日期 1 2 3 4 5 6 7 8 9 10 11 12
价格 68 69 54 64 68 64 70 67 78 62 98 87
最优秀的投资者可以购买最多4次股票,可行方案中的一种是:
日期 2 5 6 10
价格 69 68 64 62
第2行: N个数,是每天的股票价格。
|
1
2
3
|
12
68 69 54 64 68 64 70 67 78 62 98 87
|
|
1
|
4 2
|
分析:在扫描[1,i-1]寻找最优解时,如果当前解与已知最优解相同,就进行累加;如果更大,就覆盖之前的结果。对于重复方案的判断,可以比较已
知最优解的末尾和和当前解的末尾,两个价格如果相同,那么就不能进行累加,而应该选取更靠后的一个。显然靠后的价格会有不少于靠前的
价格的方案数,例如序列3,2,3,2,1:
num[2]=1,num[4]=2。
为了方便起见,可以从后往前扫描,即i-1 to 1,并用t记录最近一个最优解的price,只有小于t的price[j]才进行累加和更新。
AC 代码:
#include<iostream>
using namespace std;
int price[], dp[], num[];
int main()
{
int n, i, j, t;
cin >> n;
for (i = ; i < n; i++)
cin >> price[i];
for (i = ; i <= n; i++)
{
num[i] = ;
t = 0x3f3f3f3f; //判断是否为相同方案的变量
for (j = i - ; j >= ; j--)
if (price[j]>price[i])
{
if (dp[j] >= dp[i])
{
t = price[j];
dp[i] = dp[j] + ;
num[i] = num[j];
}
else if (dp[j] + == dp[i] && price[j] < t)
{
t = price[j];
num[i] += num[j];
}
}
}
cout << dp[n] << ' ' << num[n] << endl;;
return ;
}
不得不说一句dp是个神奇而强大的思想~~~~
动态规划---最长上升子序列问题(O(nlogn),O(n^2))的更多相关文章
- 动态规划 最长公共子序列 LCS,最长单独递增子序列,最长公共子串
LCS:给出两个序列S1和S2,求出的这两个序列的最大公共部分S3就是就是S1和S2的最长公共子序列了.公共部分 必须是以相同的顺序出现,但是不必要是连续的. 选出最长公共子序列.对于长度为n的序列, ...
- 动态规划——最长上升子序列LIS及模板
LIS定义 一个数的序列bi,当b1 < b2 < … < bS的时候,我们称这个序列是上升的.对于给定的一个序列(a1, a2, …, aN),我们可以得到一些上升的子序列(ai1 ...
- 动态规划 - 最长公共子序列(LCS)
最长公共子序列也是动态规划中的一个经典问题. 有两个字符串 S1 和 S2,求一个最长公共子串,即求字符串 S3,它同时为 S1 和 S2 的子串,且要求它的长度最长,并确定这个长度.这个问题被我们称 ...
- 动态规划 - 最长递增子序列(LIS)
最长递增子序列是动态规划中经典的问题,详细如下: 在一个已知的序列{a1,a2,...,an}中,取出若干数组组成新的序列{ai1,ai2,...,aim},其中下标i1,i2,...,im保持递增, ...
- nlog(n)解动态规划--最长上升子序列(Longest increasing subsequence)
最长上升子序列LIS问题属于动态规划的初级问题,用纯动态规划的方法来求解的时间复杂度是O(n^2).但是如果加上二叉搜索的方法,那么时间复杂度可以降到nlog(n). 具体分析参考:http://b ...
- 算法导论-动态规划(最长公共子序列问题LCS)-C++实现
首先定义一个给定序列的子序列,就是将给定序列中零个或多个元素去掉之后得到的结果,其形式化定义如下:给定一个序列X = <x1,x2 ,..., xm>,另一个序列Z =<z1,z2 ...
- 动态规划---最长公共子序列 hdu1159
hdu1159 题目要求两个字符串最长公共子序列, 状态转换方程 f[i][j]=f[i-1][j-1]+1; a[i]=b[j]时 f[i][j]=MAX{f[i-1][j],f[i][j-1] ...
- 《算法导论》读书笔记之动态规划—最长公共子序列 & 最长公共子串(LCS)
From:http://my.oschina.net/leejun2005/blog/117167 1.先科普下最长公共子序列 & 最长公共子串的区别: 找两个字符串的最长公共子串,这个子串要 ...
- 动态规划----最长递增子序列问题(LIS)
题目: 输出最长递增子序列的长度,如输入 4 2 3 1 5 6,输出 4 (因为 2 3 5 6组成了最长递增子序列). 暴力破解法:这种方法很简单,两层for循环搞定,时间复杂度是O(N2). 动 ...
随机推荐
- Android 开发笔记“程序安装包APK的制作”
资源来源:http://blog.csdn.net/qualcent/article/details/6959547 完成Android项目后,需要将程序打包成APK文件(Android Packag ...
- javascript实现的功能--二级联动
<head> <meta http-equiv="Content-Type" content="text/html;charset=UTF-8" ...
- apache域名重定向301跳转 .htaccess的写法
RewriteEngine on RewriteBase / RewriteCond %{HTTP_HOST} ^baidu.com$ [NC] RewriteRule ^(.*)$ http://w ...
- [转]android Handler使用
转 http://blog.csdn.net/new_abc/article/details/8184634 不过这个我看不懂 不知道为什么i的值可以接着增长... package com.examp ...
- perl5 第十章 格式化输出
第十章 格式化输出 by flamephoenix 一.定义打印格式二.显示打印格式三.在打印格式中显示值 1.通用的打印格式 2.格式和局域变量 3.选择值域格式 4.输出值域字符四.输出到 ...
- MBTI性格自测
这两天身边许多朋友都在测一个叫做MBTI的测试,这个测试从动力.信息收集.决策方式.生活方式四个方面评价一个人. 我发现原来在日常的交往中的差异和冲突不是没理由的,也不是别人故意要为难你,而是因为不懂 ...
- [uva 11762]Race to 1[概率DP]
引用自:http://hi.baidu.com/aekdycoin/item/be20a91bb6cc3213e3f986d3,有改动 题意: 已知D, 每次从[1,D] 内的所有素数中选择一个Ni, ...
- Android开发小记
一,下载解压adt-bundle,直接可以用来开发了二,新建android项目时不勾选创建activity,来看看如何手动创建activity1,在空项目添加class文件,选择超类为activity ...
- SQL 语言划分
从功能上划分,SQL 语言能够分为DDL,DML和DCL三大类. 1. DDL(Data Definition Language) 数据定义语言,用于定义和管理 SQL 数据库中的全部对象的语 ...
- Angular JS 学习笔记(一)
1. 菜鸟教程:http://www.runoob.com/angularjs/angularjs-tutorial.html 2. Angular JS中文网:http://www.apjs.net ...