动态规划(1)——最长子序列(LCS)问题
最长子序列问题:从中找出最长的字符序列,比如: cnblogs和belong。这两个字符串的最长子序列就是blog。
动态规划:通过分解大问题,不断的将大问题变成小问题,最终整合所有解,得出最优解(和递归有点相似,但是递归的时间复杂度太过大,通过动态规划的解决,可以将一部分的时间复杂度调整成空间复杂度)
Xm = {x1,x2,x3...xm},Yn = {y1,y2,y3,...yn},求X和Y的最长子序列。
1,假设Z = {z1,z2,..., zk}是X和Y的最长子序列,那么可以看出(解1)
- 如果xm = yn ,那么Zk-1 就是Xm-1和Yn-1的LCS(因为最后一个元素相等且已经规定Zk是Xm和Yn的LCS,所以Zk-1 自然就是Xm-1和Yn-1的LCS)
- 如果xm ≠ yn ,那么有Zk = {Xm-1,Yn}的LCS或者Zk = {Xm,Yn-1}的LCS(因为X和Y的最后一个元素不相同,所以自然最后一个元素不在LCS序列中,但是并不知道到底是哪个字符串不存在于序列中,所以这里拆分成了两个子问题)
所以通过上面的分析,可以得出状态转移方程(该方程记录的是所有状态改变的过程,即记录每个状态的过程,通过二维数组记录)
c[i,j] = 1. 0 i=0 || j =0
2. c[i-1,j-1] +1 i > 0 and j > 0 and xi = yj
3. Math.Max(c[i-1,j],c[i,j-1]) i > 0 and j > 0 and xi ≠ yj
(解释2:即始终保存的是目前最长的子串长度,通过解1的第一点可以看出如果最后一个元素相同那么LCS就是两个字符串长度-1的LCS,由于原问题比较庞大,所以现在是通过拆分原问题将它变成很多小问题来解决;解释3:同参考解释2和解1的第二点)
状态转移表如下表显示:
| i | |||||||||
| 1 | 2 | 3 | 4 | 5 | 6 | 7 | |||
| c | n | b | l | o | g | s | |||
| j | 1 | b | 0 | 0 | 1 | 1 | 1 | 1 | 1 |
| 2 | e | 0 | 0 | 1 | 1 | 1 | 1 | 1 | |
| 3 | l | 0 | 0 | 1 | 2 | 2 | 2 | 2 | |
| 4 | o | 0 | 0 | 1 | 2 | 3 | 3 | 3 | |
| 5 | n | 0 | 1 | 1 | 2 | 3 | 3 | 3 | |
| 6 | g | 0 | 1 | 1 | 2 | 3 | 4 | 4 |
代码如下:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace ConsoleApp1
{
class Program
{
public static int[,] flag = new int[,];
public static string str1 = "cnblogs";
public static string str2 = "belong";
static void Main(string[] args)
{ int[,] c = Lcs(str1, str2);
for (int i = ;i<str1.Length+;++i)
{
for(int j= ;j<str2.Length+;++j)
{
Console.Write(c[i, j]);
}
Console.Write("\n");
}
Console.WriteLine("输出LCS序列");
printLcs(str1.Length,str2.Length);
Console.ReadKey();
} public static void printLcs(int i,int j)
{
if (i == || j == ) return;
if (flag[i, j] == )
{
//Console.Write(str1[i - 1]);
printLcs(i - , j - );
Console.Write(str1[i - ]); }
else if (flag[i, j] == )
{
printLcs(i - , j); }
else
printLcs(i, j - );
} public static int[,] Lcs(string str1, string str2)
{
int len1 = str1.Length+;
int len2 = str2.Length+;
int[,] c = new int[len1, len2];
string[] sc = new string[Math.Max(len1, len2)];
for(int i = ;i<len1;i++)
{
for(int j =;j<len2;j++)
{
if (i == || j == )
c[i, j] = ;
else if (str1[i - ] == str2[j - ])
{
c[i, j] = c[i - , j - ] + ;
flag[i, j] = ;
}
else
{
c[i, j] = Math.Max(c[i, j - ], c[i - , j]);
if (c[i, j - ] <= c[i - , j])
flag[i, j] = ;
else
flag[i, j] = -;
}
}
}
for(int i = ;i<;++i)
{
for(int j=;j<;++j)
{
Console.Write(flag[i, j] + " ");
}
Console.Write("\n");
}
return c;
}
}
}
其中,完成状态转移的片段如下
int len1 = str1.Length+;
int len2 = str2.Length+;
int[,] c = new int[len1, len2];
string[] sc = new string[Math.Max(len1, len2)];
for(int i = ;i<len1;i++)
{
for(int j =;j<len2;j++)
{
if (i == || j == )
c[i, j] = ;
else if (str1[i - ] == str2[j - ])
{
c[i, j] = c[i - , j - ] + ;
flag[i, j] = ;
}
else
{
c[i, j] = Math.Max(c[i, j - ], c[i - , j]);
if (c[i, j - ] <= c[i - , j])
flag[i, j] = ;
else
flag[i, j] = -;
}
}
}
注:该片段代码只提供状态转移的过程和该问题的最长子序列的长度,若需要确定LCS的元素,则需要通过另外一个数组保存状态转移的信息(即该状态是从何而来,是从哪个数据继承过来的);flag数组代表的就是状态转移的信息,这里分成三种情况,代表着当前状态的来源,分别是:c[i-1,j-1]、c[i-1,j]、c[i,j-1]。
实现代码如下:
public static void printLcs(int i,int j)
{
if (i == || j == ) return;
if (flag[i, j] == )
{
//Console.Write(str1[i - 1]);
printLcs(i - , j - );
Console.Write(str1[i - ]); }
else if (flag[i, j] == )
{
printLcs(i - , j); }
else
printLcs(i, j - );
}
该问题通过保存状态转移的信息,然后再利用递归的方法得出结果。
代码解读:第一个 if 中,判断传入的 i 与 j (i 和 j 代表存储状态转移信息的数组的下标),倘若所有数据已经遍历完毕,那么终止递归;第二个 if 中,如果flag中保存的数据 == 0 则表示当前数据来自于 c[i-1,j-1] 所以此时递归参数就是 i-1 和 j-1,而 else if 和 else 也相应的代表另外两个来源方向。
注:由于该方法是从后到前的递归,所以Console要在函数后面,意味着先递归再输出,那么输出的结果就是从头开始,加入先输出再递归的话,那么输出的结果就是刚好相反的,比如正确的结果是 blog,那么由于错误的输出方式则造成了现在的结果是 golb。
为什么选择从后往前递归呢?第一点当然是为了方便,大家都习惯于将递归的结束条件写成与0有关,第二点是因为该数据状态的继承是从前面的数据得到的,而不是从后面的数据得到的,所以只有后面的数据可以找到前面的数据,而前面的无法预知后面的。(我猜是这样的!)
最后一点:动态规划中最重要的一步是写出该问题的状态转移方程,将问题划分成若干子问题;最核心的一步是知道要用动态规划(这不是废话吗!通常碰到类似于递归,分治的,如果不可行或者时间复杂度过于庞大的话就要考虑动态规划了)
动态规划(1)——最长子序列(LCS)问题的更多相关文章
- Poj1159 Palindrome(动态规划DP求最大公共子序列LCS)
一.Description A palindrome is a symmetrical string, that is, a string read identically from left to ...
- 算法:Common Subsequence(动态规划 Java 最长子序列)
Description A subsequence of a given sequence is the given sequence with some elements (possible non ...
- hdu1422重温世界杯(动态规划,最长子序列)
重温世界杯 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Submi ...
- nyoj17-单调递增最长子序列-(dp)
17-单调递增最长子序列 内存限制:64MB 时间限制:3000ms 特判: No通过数:125 提交数:259 难度:4 题目描述: 求一个字符串的最长递增子序列的长度如:dabdbf最长递增子序列 ...
- 【LCS,LIS】最长公共子序列、单调递增最长子序列
单调递增最长子序列 时间限制:3000 ms | 内存限制:65535 KB 难度:4 描述 求一个字符串的最长递增子序列的长度如:dabdbf最长递增子序列就是abdf,长度为4 输入 ...
- 动态规划:ZOJ1074-最大和子矩阵 DP(最长子序列的升级版)
To the Max Time Limit:1 Second Memory Limit:32768 KB Problem Given a two-dimensional array of po ...
- nyoj 17-单调递增最长子序列 && poj 2533(动态规划,演算法)
17-单调递增最长子序列 内存限制:64MB 时间限制:3000ms Special Judge: No accepted:21 submit:49 题目描述: 求一个字符串的最长递增子序列的长度 如 ...
- 动态规划 最长公共子序列 LCS,最长单独递增子序列,最长公共子串
LCS:给出两个序列S1和S2,求出的这两个序列的最大公共部分S3就是就是S1和S2的最长公共子序列了.公共部分 必须是以相同的顺序出现,但是不必要是连续的. 选出最长公共子序列.对于长度为n的序列, ...
- 最长递增子序列问题 nyoj 17单调递增最长子序列 nyoj 79拦截导弹
一, 最长递增子序列问题的描述 设L=<a1,a2,…,an>是n个不同的实数的序列,L的递增子序列是这样一个子序列Lin=<aK1,ak2,…,akm>,其中k1< ...
随机推荐
- js中的scrollTop、offsetTop、clientTop
scrollHeight:获取对象可滚动的高度. scrollWidth:获取对象可滚动的宽度. scrollTop:获取对象最顶端与对象可见区域最顶端的距离. scrollLeft:获取对象左边界与 ...
- 如何在 Linux 中添加一块大于 2TB 的新磁盘?
你有没有试过使用 fdisk 对大于 2TB 的硬盘进行分区,并且纳闷为什么会得到需要使用 GPT 的警告? 是的,你看到的没错.我们无法使用 fdisk 对大于 2TB 的硬盘进行分区. 在这种情况 ...
- WIFI Manager
Vistumbler - wifi managerhttps://www.vistumbler.net/downloads.htmlhttps://github.com/RIEI
- 访问类的私有属性(RTTI和模仿类2种方法)
如何访问类的私有属性? 下面以 TPathData 为例,它有一个私有属性 PathData,储存了每一个曲线点,但一般无法修改它,需要利用下面方法,才能访问修改(若有更好的方法,歡迎分享): 一.利 ...
- QT多个UI文件加入一个项目
这样可在多个UI界面上进行分部开发.避免都在一个UI下太凌乱…… 在网上找了一些资料,很少有介绍这方面的,以及类似这样项目的源码. 看 一些基本控件的使用时,想到了一种方法:使用gridLayout控 ...
- Qemu搭建ARM vexpress开发环境(二)----通过u-boot启动Linux内核
Qemu搭建ARM vexpress开发环境(二)----通过u-boot启动Linux内核 标签(空格分隔): Qemu ARM Linux 在上文<Qemu搭建ARM vexpress开发环 ...
- ZooKeeper 系列(三)—— Zookeeper常用 Shell 命令
一.节点增删改查 1.1 启动服务和连接服务 1.2 help命令 1.3 查看节点列表 1.4 新增节点 1.5 查看 ...
- Spark —— 高可用集群搭建
一.集群规划 这里搭建一个3节点的Spark集群,其中三台主机上均部署Worker服务.同时为了保证高可用,除了在hadoop001上部署主Master服务外,还在hadoop002和hadoop00 ...
- 浅谈block, inline和inline-block的区别
block 块元素 inline 内联元素 常见的块元素有:div, p, h1~h6, table, form, ol, ul等 常见的内联元素有:span, a, strong, em, l ...
- java内存管理机制剖析(一)
最近利用工作之余学习研究了一下java的内存管理机制,在这里记录总结一下. 1-1.java内存区域 当java程序运行时,java虚拟机会将内存划分为若干个不同的数据区域,这些内存区域创建和销毁的时 ...