LIS LCS 最长上升子序列 最长公共子序列 ...
最长上升子序列,问题定义:http://blog.csdn.net/chenwenshi/article/details/6027086
代码:
public static void getData( char[] L )
{
int len = L.length;
int[] f = new int[len];
String[] res = new String[len];
for( int i = ; i < len; i++ )
{
f[i] = ;
res[i] = "" + L[i];
for( int j = ; j < i; j++ )
{
if( (L[j] - '') < (L[i]-'') && f[j] + > f[i] )
{
f[i] = f[j] + ;
res[i] = res[j] + " " + L[i];
}
}
/* 下标连续递增子序列
if( (L[i-1] - '0') == (L[i]-'0') - 1 )
{
f[i] = f[i-1] + 1;
res[i] = res[i-1] + " " + L[i];
}
/**/
}
// System.out.println( f[ len - 1 ]);
int max = f[];
int maxK = ;
for( int k = ; k < len; k++ )
{
if( f[k] > max )
{
max = f[k];
maxK = k;
}
}
System.out.println( f[ maxK ]);
for( int k = ; k < len; k++ )
{
if( f[k] == max )
{
System.out.println( res[k] );
}
}
}
最大和子序列(最大和连续子序列)。MaxSum[i] 表示以i结尾的有最大和的连续子序列之和。MaxSum[i] = Max{ MaxSum[i-1] + A[i], A[i]}; 以下代码没有设置数组MaxSum[]。
题目描述:http://blog.csdn.net/hs794502825/article/details/7956730
public static void getMax( String[] L )
{
int thisSum = 0, maxSum = 0;
int begin=0,end = 0;
int max = 0; //String[] res = new String[L.length];
//res[0] = "" + L[0];
for( int j = 1; j < L.length; j++ )
{
thisSum += Integer.parseInt( L[j] );
if( thisSum > maxSum )
{
maxSum = thisSum;
//max = j;
end = j;
}
if( thisSum < 0 )
{
thisSum = 0;
begin = j+1;
}
//res[j] = res[j-1] + " " + L[j];
//end = j;
} System.out.println( maxSum ); for( int k = begin; k <= end; k++ )
{
System.out.print( L[k] + " " );
} }
L1与L2的最长公共子序列:
public static void getLCS( char[] x, char[] y )
{
int m = x.length;
int n = y.length;
int[][] c = new int[m+1][n+1];
int[][] b = new int[m+1][n+1];
for( int row = 0; row < m; row ++ )
{
int i = row + 1;
for( int col = 0; col < n; col ++ )
{
int j = col + 1;
if( x[row] == y[col] )
{
c[i][j] = c[i-1][j-1] + 1;
b[i][j] = 0;
}
else if( c[i-1][j] > c[i][j-1] )
{
c[i][j] = c[i-1][j];
b[i][j] = 1;
}
else if( c[i-1][j] < c[i][j-1] )
{
c[i][j] = c[i][j-1];
b[i][j] = 2;
}
else
{
c[i][j] = c[i][j-1];
b[i][j] = 3;
}
}
} int i = m;
int j = n;
Stack< Character > stack = new Stack< Character >();
while( (i != 0) && ( j != 0) )
{
if( b[i][j] == 0 )
{
//System.out.println( x[m] );
stack.push(x[i-1]);
i -= 1;
j -= 1;
}
else if( b[i][j] == 1 )
{
i -= 1;
}
else
{
j -= 1;
}
}
while( !stack.isEmpty() )
{
System.out.print( stack.pop() );
}
}
public static void main(String[] args) {
// TODO Auto-generated method stub
String a = "abcdeabcdefg";
String b = "acdeafdc";
getLCS( a.toCharArray(), b.toCharArray() );
}
装信封问题:
N个信封,给出宽、高。小信封装在大信封里,求这些信封最多能嵌套的层数。
第一行为N,接下来N行每行两个整数,分别表示宽与高。
4
5 4
6 4
6 7
2 3
这个问题可以转化为最长递增子序列问题。首先按照宽度将N个信封由小到大排序,当宽度相同时,按高度从大到小排序:2 3, 5 4, 6 7, 6 4。防止了当宽度相同时,高度小的嵌套进高度大的信封。代码如下:
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std; bool cmp( const pair<int,int>& x, const pair<int,int>& y )
{
return x.first != y.first ? x.first < y.first : x.second > y.second;
}
int findMax( vector< pair< int, int > > e )
{
int n = e.size();
if( n == )
{
return ;
} sort( e.begin(), e.end(), cmp ); vector<int> dp(n,);
for( int i = ; i < n; i++ )
{
//dp[i] = 1;
for( int j = ; j < i; j++ )
{
if( ( dp[i] < dp[j]+ ) && ( e[i].second > e[j].second ) && ( e[i].first > e[j].first ) )
{
dp[i] = dp[j]+;
}
}
}
int maxv = ;
for( int i = ; i < n; i++ )
{
if( dp[i] > maxv )
{
maxv = dp[i];
}
} // O(nlogn):
// http://blog.csdn.net/shuangde800/article/details/7474903
// http://www.jiuzhang.com/solutions/russian-doll-envelopes/
// 当有多个子序列s1,s2,,,内元素数为均为k个时,选择si,si的第k个元素比其他s1,s2...的第k个元素都小
// 数组d[k]即表示,当上升子序列元素个数为k时,si中第k个元素的值
vector< int > d( n+, );
int len = ;
for( int k = ; k < n; k++ )
{
if( d[len] < e[k].second )
{
d[++len] = e[k].second;
}
else
{
//需要找到比d[mid]大的第一个位置,即插入数据的位置 //d数组中的元素不会重复,用二分插入排序查找插入位置就可以
//index: 1 2 3 4 5
//value: a a a a b
//区别:
//查找下界的二分查找,查找a时返回第一个a的位置1
//二分插入排序查找插入位置,插入a时,返回最后一个a的下一个位置6 //stl中binary_search函数功能是查看某一值在一个已经排好序的序列中是否存在,
//当存在时返回true,否则返回false //如果所要查找的元素只有一个,那么lower_bound()返回了这个元素的地址,不是下标
//如果相同元素出现了多次,那么lower_bound()找到了第一个所找元素的地址
//即:返回第一个大于或等于val的元素的地址
//upper_bound返回第一个大于val元素的地址
//算法竞赛入门 p.125 int left=, right=len, mid, j = -;
while( left <= right )
{
mid = ( left + right )/;
//当 d[mid] 与 e[k].second(待插入数值)相等时,left也要置为mid+1,
//这也保证也二分插入排序是稳定的
if( d[mid] > e[k].second )
{
right = mid - ;
}
else
{
left = mid + ;
}
} // left; 待插入数据位置,第一个待插入数字大的位置
d[ left ] = e[k].second;
}
}
cout << len << "+++++++" << endl; return maxv;
} int main()
{
vector< pair< int, int > > e;
int n;
cin >> n;
int a, b;
for( int i = ; i < n; i++ )
{
cin >> a >> b;
e.push_back( make_pair<int,int>( a, b ) );
} cout << findMax( e );
return ;
}
在O(nlogn)的实现中,还引申出了二分查找的上下界问题。
另外,当要求最长不下降子序列时,这篇文章给出了很好的分析:http://www.cnblogs.com/itlqs/p/5743114.html
LIS LCS 最长上升子序列 最长公共子序列 ...的更多相关文章
- DP_最长公共子序列/动规入门
学自:https://open.163.com/movie/2010/12/L/4/M6UTT5U0I_M6V2U1HL4.html 最长公共子序列:(本文先谈如何求出最长公共子序列的长度,求出最长公 ...
- 最长公共子序列(LCS)和最长递增子序列(LIS)的求解
一.最长公共子序列 经典的动态规划问题,大概的陈述如下: 给定两个序列a1,a2,a3,a4,a5,a6......和b1,b2,b3,b4,b5,b6.......,要求这样的序列使得c同时是这两个 ...
- 算法设计 - LCS 最长公共子序列&&最长公共子串 &&LIS 最长递增子序列
出处 http://segmentfault.com/blog/exploring/ 本章讲解:1. LCS(最长公共子序列)O(n^2)的时间复杂度,O(n^2)的空间复杂度:2. 与之类似但不同的 ...
- 最长递增子序列(lis)最长公共子序列(lcs) 最长公共上升子序列(lics)
lis: 复杂度nlgn #include<iostream> #include<cstdio> using namespace std; ],lis[],res=; int ...
- 最长公共子序列-LCS问题 (LCS与LIS在特殊条件下的转换) [洛谷1439]
题目描述 给出1-n的两个排列P1和P2,求它们的最长公共子序列. 输入 第一行是一个数n, 接下来两行,每行为n个数,为自然数1-n的一个排列. 输出 一个数,即最长公共子序列的长度 输入样例 5 ...
- 最长上升子序列(LIS)与最长公共子序列(LCS)
1.LIS : 给定一个序列,求它的最长上升子序列(n<=2000) 第一种 O(n^2): dp[i] 为以i为开头的最长上升子序列长度 code1: #include<cstdio&g ...
- 最长连续公共子序列(LCS)与最长递增公共子序列(LIS)
最长公共子序列(不连续) 实际问题中也有比较多的应用,比如,论文查重这种,就是很实际的一个使用方面. 这个应该是最常见的一种了,不再赘述,直接按照转移方程来进行: 按最普通的方式就是,直接构造二维矩阵 ...
- 最长公共子序列(LCS)、最长递增子序列(LIS)、最长递增公共子序列(LICS)
最长公共子序列(LCS) [问题] 求两字符序列的最长公共字符子序列 问题描述:字符序列的子序列是指从给定字符序列中随意地(不一定连续)去掉若干个字符(可能一个也不去掉)后所形成的字符序列.令给定的字 ...
- 【线型DP模板】最上上升子序列(LIS),最长公共子序列(LCS),最长公共上升子序列(LCIS)
BEGIN LIS: 一个数的序列bi,当b1 < b2 < … < bS的时候,我们称这个序列是上升的.对于给定的一个序列(a1, a2, …, aN),我们可以得到一些上升的子序 ...
- 【LCS,LIS】最长公共子序列、单调递增最长子序列
单调递增最长子序列 时间限制:3000 ms | 内存限制:65535 KB 难度:4 描述 求一个字符串的最长递增子序列的长度如:dabdbf最长递增子序列就是abdf,长度为4 输入 ...
随机推荐
- Kubernetes 1.8.x 全手动安装教程----转自Kubernetes中文社区(部分内容根据实验环境做了些修改,特此感谢Kubernetes中文社区)
Kubernetes 提供了许多云端平台与操作系统的安装方式,本章将以全手动安装方式来部署,主要是学习与了解 Kubernetes 创建流程.若想要了解更多平台的部署可以参考 Picking the ...
- 一张图知道HTML5布局(图)
- Android构建项目时出现的小bug们(2018年5月19日19:31:20)
问题详情 Error:Execution failed for task ':app:preDebugAndroidTestBuild'. > Conflict with dependency ...
- 5分钟K线图压力线买点怎么看?
某开盘后底开一直呈现形成了一个长时间的箱体振荡的走势,K线在底位振荡时,其波动底点总是在不断抬高的话,这种走势说明有资金在场中积极运作,正是由于资金悄然建仓导致了底点慢慢抬高的走势,在底点不断抬高时, ...
- C++ 设置透明背景图片
背景: 有两个图片,一个是目标背景图片, 一个是带有自身背景色彩的彩色图片 先将这彩色图片绘制到目标背景图片中, 这一步通过BITBLT就可实现. 但实 ...
- typora中文版官方免费快速下载以及Markdown的一些常用语法、Java知识点
typora下载 链接:https://某度云盘的域名/s/1geD1APxnyV3gogYW3E08bQ 密码:8fdp 把某度云盘的域名进行替换 1.标题 # 标题1 ## 标题2 ### 标题3 ...
- JAVA回文
package huiwen; import java.util.Scanner; public class Huiwen { public static void main(String[] arg ...
- 内核 platform_get_resource() 函数解析
struct resource *platform_get_resource(struct platform_device *dev, unsigned int type, un ...
- linux配置虚拟域名
linux环境下面配置虚拟主机域名 第一步:在root目录下面(即根目录)ls(查看文件)cd进入etc目录find hosts文件vi hosts 打开hosts文件并进行编辑在打开的文件最下面添加 ...
- ELK 日志学习
一.Elasticsearch 安装(所有的版本均使用5.5.0 ,且版需要相同 logstash \ kibana \ filebeat ) 官网下载地址:https://www.elastic.c ...