动态规划--最长公共子序列

这篇具有很好参考价值的文章主要介绍了动态规划--最长公共子序列。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

动态规划算法思想

动态规划算法与分治法类似,其基本思想也是将待求解问题分解成若干个子问题﹐即将大规模变成小规模,先求解子问题,然后从这些子问题的解得到原问题的解。与分治法不同的是﹐适合于用动态规划法求解的问题,经分解得到的子问题往往不是互相独立的。他们之间有关系,所以用一个表来记录所有已解决的子问题的答案。不管该子问题以后是否被用到,只要它被计算过,就将其结果填人表(可以是二维,一维数组,或者是变量)中。这就是动态规划法的基本思想。

最长公共子序列

题目

给定两个序列,返回这两个序列的最长公共子序列的长度。
一个序列的子序列是指这样一个新的序列:它是由原序列在不改变字符的相对顺序的情况下删除某些字符(也可以不删除任何字符)后组成的新序列。
两个序列的公共子序列是这两个序列所共同拥有的子序列。
最长公共子序列问题:给定两个序列X={x1,x2,…xm}和Y={ y1,y2,…yn},找出X和Y的最长公共子序列。
动态规划算法可有效地解此问题。下面按照动态规划算法设计的各个 步骤设计解此问题的有效算法。

最优解结构性质

我们假设Z是X和Y的最长公共子序列,该假设的目的是使该问题的规模不断缩小
最优解结构性质

X={x1,x2,...xm}  m
Y={y1,y2,...yn}  n
Z={z1,z2,...zk}  k

三种情况

  • xm== yn==zk
    • Zk-1=> Xm-1:Yn-1
  • xm!=yn && zk!=xm
    • Zk=>Xm-1:Yn
  • xm!=yn && zk!=yn
    • Zk=>Xm:Yn-1

c[i][j] =>k:
i是X序列的长度,j是Y序列的长度,c代表序列X有i个元素时,序列Y有j个元素时,最长的公共子序列的长度

注意:i不是点值,i是区域值,从1 2到i,j不是点值,j是区域值,从1 2到j

状态转移方程

分治策略的核心,规模减小到0,如果最后一个值相等,就减小一个规模,如果不相等,有一个max的概念,c[i][j]=c[i-1][j-1]+1可以写成c[i-1][j-1]=c[i][j]-1更好理解,减小规模

  • i == 0 || j ==0
    • c[i][j]=0
  • i>0&&j>0 x[i]==y[j]
    • c[i][j]=c[i-1][j-1]+1
  • i>0&&j>0 x[i]!=y[j]
    • c[i][j]=max(c[i-1][j],c[i][j-1])

所以递归方程为
最长公共子序列动态规划算法,算法,动态规划,算法

递归实现

递归实现是自顶向底,等到递归到底部的时候,即终止条件为任意一个长度为0,说明公共子序列的长度一定为0,再依次回溯,最终回溯的答案就是最长公共子序列的长度

核心函数

int sum = 0;
int LCSLength(const char* X, const char* Y, int i, int j) {
	sum += 1;
	if (i == 0 || j == 0)return 0;
	if (X[i] == Y[j]) {
		return LCSLength(X, Y, i - 1, j - 1) + 1;
	}
	else {
		int xs = LCSLength(X, Y, i - 1, j);
		int ys = LCSLength(X, Y, i, j - 1);
		if (xs >= ys)return xs;
		else return ys;
	}
}

测试

测试的话,我们求ABCBDAB和BDCABA的最长公共子序列
易知为BCBA ,该长度为4

#include<iostream>
#include<vector>
using namespace std;
int main() {
	const char* X = { "#ABCBDAB" };
	const char* Y = { "#BDCABA" };
	int xm = strlen(X) - 1;
	int yn = strlen(Y) - 1;
	int maxlen = LCSLength(X, Y, xm, yn);
	cout << maxlen << endl;
	cout << sum << endl;
	return 0;
}

测试结果

该函数递归调用次数152,指数增长,有三个情况,一个递归函数里面嵌套了三个递归函数。比如二叉树中序遍历,每次有两个情况,一个递归函数里面嵌套了两个函数
最长公共子序列动态规划算法,算法,动态规划,算法

非递归实现(画表)

递归实现是从顶部向下考虑,中间有大量的重复计算,而非递归实现即动态规划,就是从底部向上画表,画到最后,结果就出来了
最长公共子序列动态规划算法,算法,动态规划,算法

核心函数

所以用填表的方法,先初始化第一行和第一列,然后一行一行的填表,根据动态规划表达式,每一个数的填写,依赖,前一个数,上一个数,左上的数字,所以每一行的填写依赖上一行
大致框架就是双层for循环,用X序列的每一个字符依次和Y序列的每一个字符


int NiceLCSLength(const char* X, const char* Y, int m, int n, vector<vector<int> >& c) {
	for (int i = 0; i <= m; i++)c[i][0] = 0;
	for (int i = 0; i <= n; i++)c[0][i] = 0;
	for (int i = 1; i <= m; i++) {
		for (int j = 1; j <= n; j++) {
			if (X[i] == Y[j]) {
				c[i][j] = c[i - 1][j - 1] + 1;
			}else  if (c[i - 1][j] >= c[i][j - 1]) {
				c[i][j] = c[i - 1][j];
			}else{
				c[i][j] = c[i][j - 1];
			}
		}
	}
	return c[m][n];
}

测试

#include<iostream>
#include<vector>
using namespace std;
void PrintVector(vector<vector<int> >& c) {
	int m = c.size();
	for (int i = 0; i < m;i++) {
		for (int j = 0;j<c[i].size();j++) {
			printf("%5d", c[i][j]);
		}
		printf("\n");
	}
}
int NiceLCSLength(const char* X, const char* Y, int m, int n, vector<vector<int> >& c) {
	for (int i = 0; i <= m; i++)c[i][0] = 0;
	for (int i = 0; i <= n; i++)c[0][i] = 0;
	for (int i = 1; i <= m; i++) {
		for (int j = 1; j <= n; j++) {
			if (X[i] == Y[j]) {
				c[i][j] = c[i - 1][j - 1] + 1;
			}else  if (c[i - 1][j] >= c[i][j - 1]) {
				c[i][j] = c[i - 1][j];
			}else{
				c[i][j] = c[i][j - 1];
			}
		}
	}
	return c[m][n];
}
int main() {
	const char* X = { "#ABCBDAB" };
	const char* Y = { "#BDCABA" };
	int xm = strlen(X) - 1;
	int yn = strlen(Y) - 1;
	vector<vector<int> >c(xm + 1, vector<int>(yn + 1,0));
	int maxlen = NiceLCSLength(X, Y, xm, yn,c);
	cout << maxlen << endl;
	PrintVector(c);
	return 0;
}

测试结果

最长公共子序列动态规划算法,算法,动态规划,算法

求出具体的子序列

标记方向数组

对于如何将具体的最长公共子序列求出来呢
我们再定义一个二维数组,记录2是行减一,3是列减一,1是行列都减一

int NiceLCSLength(const char* X, const char* Y, int m, int n, vector<vector<int> >& c, vector<vector<int> >& s) {
	for (int i = 0; i <= m; i++)c[i][0] = 0;
	for (int i = 0; i <= n; i++)c[0][i] = 0;
	for (int i = 1; i <= m; i++) {
		for (int j = 1; j <= n; j++) {
			if (X[i] == Y[j]) {
				c[i][j] = c[i - 1][j - 1] + 1;
				s[i][j] = 1;
			}
			else  if (c[i - 1][j] >= c[i][j - 1]) {
				c[i][j] = c[i - 1][j];
				s[i][j] = 2;
			}else{
				c[i][j] = c[i][j - 1];
				s[i][j] = 3;
			}
		}
	}
	return c[m][n];
}

最长公共子序列动态规划算法,算法,动态规划,算法
最长公共子序列动态规划算法,算法,动态规划,算法

回溯出具体序列

当i或者j一个为0时就说明搜索结束,依次回溯序列BCBA

void LCS(int i, int j, const char* X, vector<vector<int> >& s) {
	if (i == 0 || j == 0)return;
	if (s[i][j] == 1) {
		LCS(i - 1, j - 1, X, s);
		printf("%5c", X[i]);
	}
	else if (s[i][j] == 2) {
		LCS(i - 1, j, X, s);
	}
	else {
		LCS(i, j - 1, X, s);
	}
}

最长公共子序列动态规划算法,算法,动态规划,算法文章来源地址https://www.toymoban.com/news/detail-756018.html

到了这里,关于动态规划--最长公共子序列的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【算法】力扣【动态规划,LCS】1143. 最长公共子序列

    1143. 最长公共子序列 本文是对 LCS 这一 动态规划 模型的整理,以力扣平台上的算法题1143:最长公共子序列为模板题进行解析。 该题目要求计算两个字符串的最长公共子序列(Longest Common Subsequence,简称LCS)的长度。字符串的子序列是指在不改变字符顺序的情况下,通过删去

    2024年01月17日
    浏览(43)
  • 算法套路十五——动态规划求解最长公共子序列LCS

    给定两个字符串 text1 和 text2,返回这两个字符串的最长 公共子序列 的长度。如果不存在 公共子序列 ,返回 0 。 一个字符串的 子序列 是指这样一个新的字符串:它是由原字符串在不改变字符的相对顺序的情况下删除某些字符(也可以不删除任何字符)后组成的新字符串。

    2024年02月04日
    浏览(40)
  • 算法分析:C语言实现动态规划之最长公共子序列

    最长公共子序列问题:          下面的简单问题说明了动态规划的基本原理。在字母表一∑上,分别给出两个长度为n和m的字符串A和B,确定在A和B中最长公共子序列的长度。这里,A = a₁a₂...an。的子序列是一个形式为a₁ka₂k...aik的字符串,其中每个i都在1和n之间,并且

    2023年04月21日
    浏览(25)
  • python数据结构与算法-动态规划(最长公共子序列)

    一个序列的子序列是在该序列中删去若干元素后得 到的序列。 例如:\\\"ABCD”和“BDF”都是“ABCDEFG”的子序列。 最长公共子序列(LCS) 问题: 给定两个序列X和Y,求X和Y长度最大的公共子字列。 例:X=\\\"ABBCBDE”Y=\\\"DBBCDB”LCS(XY)=\\\"BBCD\\\" 应用场景:字符串相似度比对 (1)问题思考 思考: 暴

    2024年02月08日
    浏览(39)
  • 算法分析 | 动态规划算法设计之最长公共子序列 C语言版

    声明:凡代码问题,欢迎在评论区沟通。承蒙指正,一起成长! 目录 一、实验内容与要求  二、概要设计 三、直接上代码      四、输入数据及运行结果   内容:最长公共子序列 ·若给定序列X={x1,x2,…,xm},则另一序列Z={z1,z2,…,zk},是X的子序列是指存在一个严格递增下标序

    2024年02月02日
    浏览(36)
  • 9.动态规划——4.最长公共子序列(动态规划类的算法题该如何解决?)

    设最长公共子序列 d p [ i ] [ j ] dp[i][j] d p [ i ] [ j ] 是 S 1 S_1 S 1 ​ 的前 i i i 个元素,是 S 2 S_2 S 2 ​ 的前 j j j 个元素,那么有: 若 S 1 [ i − 1 ] = = S 2 [ i − 1 ] S_1[i-1]==S_2[i-1] S 1 ​ [ i − 1 ] == S 2 ​ [ i − 1 ] ,那么 d p [ i ] [ j ] = d p [ i − 1 ] [ j − 1 ] + 1 dp[i][j]=dp[i-1][j-1]+1 d p [

    2024年04月11日
    浏览(34)
  • 算法 DAY52 动态规划10 1143.最长公共子序列 1035.不相交的线 53. 最大子数组和

    本题和动态规划:718. 最长重复子数组 (opens new window)区别在于这里不要求是连续的了 1、dp数组 dp[i][j]:长度为[0, i - 1]的字符串text1与长度为[0, j - 1]的字符串text2的最长公共子序列为dp[i][j] 2、递推公式 因为不强调是连续的,当前dp[i][j] 就有三种路径可以选:dp[i-1][j] dp[i][j-1]

    2024年02月03日
    浏览(49)
  • 动态规划——最长公共子序列

    先来讲解以下什么是最长公共子序列。最长公共子序列不是最长相同字符串,有点相似但不一样,来举个简单的例子,有字符串s1=bcdea,s2=abce,最长相同字符串是bc,最大公共部分是2;而最长公共子序列则是bce,最大公共部分是3。可以看出,公共子序列不需要连续相等,有相

    2023年04月19日
    浏览(36)
  • 动态规划--最长公共子序列

    动态规划算法与分治法类似,其基本思想也是将待求解问题分解成若干个子问题﹐ 即将大规模变成小规模 ,先求解子问题,然后从这些子问题的解得到原问题的解。与分治法不同的是﹐适合于用动态规划法求解的问题,经分解得到的子问题往往不是互相独立的。 他们之间有关系

    2024年02月04日
    浏览(49)
  • 【动态规划】最长公共子序列(Java)

    给定两个字符串 text1 和 text2,返回这两个字符串的最长 公共子序列 的长度。如果不存在 公共子序列 ,返回 0 。 一个字符串的 子序列 是指这样一个新的字符串:它是由原字符串在不改变字符的相对顺序的情况下删除某些字符(也可以不删除任何字符)后组成的新字符串。

    2024年01月18日
    浏览(57)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包