两句闲话

  得到两个序列的最长公共子序列(LCS)是个经典问题,使用动态规划,实现起来并不难。

  一般来说,我们只是输出一个LCS。但是,老师布置的作业是输出所有的LCS。

解法

  按照一般的方法,我们首先得到一个矩阵,然后从矩阵的右下角开始回溯。回溯时,我们选择较大的数字,以向左,或向上,或向左上。但当数字相等时,我们往往会随便向某一个方向回溯,这样的话,我们就只会得到一个LCS。因此,很容易想到,所有的LCS会构成一棵树,我们只需要对这棵树进行先序遍历,就可得到所有的LCS。

  

  代码如下

#python 3.5

class LCS_naive:
"""
最长公共子序列:
通过动态规划,得到矩阵D,
并从矩阵D中读出一个最长公共子序列
不支持读出所有的LCS
"""
def __init__(self):
self.matrix=[[]] def init(self,str1,str2):
self.str1=str1
self.str2=str2
self.len1=len(str1)
self.len2=len(str2)
self.matrix=[[0 for i in range(self.len2+1)]for j in range(self.len1+1)] def _get_matrix(self):
"""通过动态规划,构建矩阵"""
for i in range(self.len1):
for j in range(self.len2):
if self.str1[i]==self.str2[j]:
self.matrix[i+1][j+1]=self.matrix[i][j]+1
else:
self.matrix[i+1][j+1]=max(self.matrix[i][j+1],self.matrix[i+1][j]) def _matrix_show(self,matrix):
"""展示通过动态规划所构建的矩阵"""
print ("----matrix-----")
print (" "," ",end=" ")
for ch in self.str2:
print (ch,end=" ")
print ()
for i in range(len(matrix)):
if i>0: print (self.str1[i-1],end=" ")
else: print (" ",end=" ")
for j in range(len(matrix[i])):
print (matrix[i][j],end=" ")
print ()
print ("---------------") def _get_one_lcs_from_matrix(self):
i=len(self.matrix)-1
if i==0:
print ("matrix is too small")
return
j=len(self.matrix[0])-1
res=[]
while not (i==0 or j==0):
if self.str1[i-1]==self.str2[j-1]:
res.append(self.str1[i-1])
i-=1
j-=1
else:
if self.matrix[i-1][j]>self.matrix[i][j-1]:
i=i-1
else:
j=j-1
return "".join(res[::-1]) def get_lcs(self):
self._get_matrix()
self._matrix_show(self.matrix)
lcs=self._get_one_lcs_from_matrix()
print (lcs) class LCS(LCS_naive):
"""
继承自LCS_naive
增加获取所有LCS的支持
"""
def __init__(self):
LCS_naive.__init__(self) def _get_all_lcs_from_matrix(self):
self._pre_travesal(self.len1,self.len2,[]) def _pre_travesal(self,i,j,lcs_ted):
if i==0 or j==0:
print ("".join(lcs_ted[::-1]))
return
if self.str1[i-1]==self.str2[j-1]:
lcs_ted.append(self.str1[i-1])
self._pre_travesal(i-1,j-1,lcs_ted)
else:
if self.matrix[i-1][j]>self.matrix[i][j-1]:
self._pre_travesal(i-1,j,lcs_ted)
elif self.matrix[i-1][j]<self.matrix[i][j-1]:
self._pre_travesal(i,j-1,lcs_ted)
else:
###### 分支
self._pre_travesal(i-1,j,lcs_ted[:])
self._pre_travesal(i,j-1,lcs_ted) def get_lcs(self):
self._get_matrix()
self._matrix_show(self.matrix)
self._get_all_lcs_from_matrix() l=LCS()
l.init("ABCBDAB","BDCABA")
l.get_lcs()

输出结果

----matrix-----
B D C A B A
0 0 0 0 0 0 0
A 0 0 0 0 1 1 1
B 0 1 1 1 1 2 2
C 0 1 1 2 2 2 2
B 0 1 1 2 2 3 3
D 0 1 2 2 2 3 3
A 0 1 2 2 3 3 4
B 0 1 2 2 3 4 4
---------------
BCBA
BCAB
BDAB

  

[python] 获得所有的最长公共子序列的更多相关文章

  1. 最长公共子序列python实现

    最长公共子序列是动态规划基本题目,以下依照动态规划基本步骤解出来. 1.找出最优解的性质,并刻划其结构特征 序列a共同拥有m个元素,序列b共同拥有n个元素,假设a[m-1]==b[n-1],那么a[: ...

  2. 用Python计算最长公共子序列和最长公共子串

    如何用Python计算最长公共子序列和最长公共子串 1. 什么是最长公共子序列?什么是最长公共子串? 1.1. 最长公共子序列(Longest-Common-Subsequences,LCS) 最长公 ...

  3. python实现查找最长公共子序列

    #!/usr/bin/python # -*- coding: UTF-8 -*- worlds = ['fosh','fort','vista','fish','hish','hello','ohd ...

  4. 【python】Leetcode每日一题-最长公共子序列

    [python]Leetcode每日一题-最长公共子序列 [题目描述] 给定两个字符串 text1 和 text2,返回这两个字符串的最长 公共子序列 的长度.如果不存在 公共子序列 ,返回 0 . ...

  5. 用python实现最长公共子序列算法(找到所有最长公共子串)

    软件安全的一个小实验,正好复习一下LCS的写法. 实现LCS的算法和算法导论上的方式基本一致,都是先建好两个表,一个存储在(i,j)处当前最长公共子序列长度,另一个存储在(i,j)处的回溯方向. 相对 ...

  6. 【ZH奶酪】如何用Python计算最长公共子序列和最长公共子串

    1. 什么是最长公共子序列?什么是最长公共子串? 1.1. 最长公共子序列(Longest-Common-Subsequences,LCS) 最长公共子序列(Longest-Common-Subseq ...

  7. python 回溯法 子集树模板 系列 —— 14、最长公共子序列(LCS)

    问题 输入 第1行:字符串A 第2行:字符串B (A,B的长度 <= 1000) 输出 输出最长的子序列,如果有多个,随意输出1个. 输入示例 belong cnblogs 输出示例 blog ...

  8. 动态规划1——最长递增子序列、最长公共子序列、最长公共子串(python实现)

    目录 1. 最长递增序列 2. 最长公共子序列 3. 最长公共子串 1. 最长递增序列 给定一个序列,找出其中最长的,严格递增的子序列的长度(不要求连续). 解法一:动态规划 通过一个辅助数组记录每一 ...

  9. [Python]最长公共子序列 VS 最长公共子串[动态规划]

    前言 由于原微软开源的基于古老的perl语言的Rouge依赖环境实在难以搭建,遂跟着Rouge论文的描述自行实现. Rouge存在N.L.S.W.SU等几大子评估指标.在复现Rouge-L的函数时,便 ...

随机推荐

  1. SSO 单点登录的实现原理

    单点登录SSO(Single Sign On)说得简单点就是在一个多系统共存的环境下,用户在一处登录后,就不用在其他系统中登录,也就是用户的一次登录能得到其他所有系统的信任.单点登录在大型网站里使用得 ...

  2. 给IT男推荐一款车

    标题是为了吸引人,其实这里要推荐的不是汽车而是自行车,确切的说是电动自行车(也有叫锂电自行车),见下图! (备注:淘宝截图,有便宜的2K就能搞定,好点的像这一款3K吧!) 码农这个词真不是瞎叫的,做这 ...

  3. linux(6/17)--文件打包上传和下载

    tar命令 命令功能 用来压缩和解压文件 命令格式 tar[必要参数][选择参数][文件] tar打包工具 -f ##指定生成包的名字,建议 -f单独写成一个参数 --delete filename ...

  4. 【Head First Servlets and JSP】笔记21:从有脚本到无脚本

    可以建立多态的bean引用吗 使用type,但没有class scope属性默认为“page” 从有脚本到无脚本 1.快速搭建一个测试环境:输入用户名,返回“Hello, 用户名” index.htm ...

  5. promise两个参数的具体作用

    Promise通常配合then方法来链式的使用,then方法里面第一个回调函数表示成功状态,也就是resolve通过.then调用,第二个是失败状态-reject通过.Cath调用,如果默认写一个参数 ...

  6. CommStringLib

    #include "syswatcher/CommStringLib.h" //#include "String16.h" #undef LOG_TAG #de ...

  7. Generate parentheses,生成括号对,递归,深度优先搜索。

    问题描述:给n对括号,生成所有合理的括号对.比如n=2,(()),()() 算法思路:利用深度优先搜索的递归思想,对n进行深度优先搜索.边界条件是n==0:前面电话号组成字符串也是利用dfs. pub ...

  8. lucene学习-3 - 代码重构

    内容就是标题了.是要重构下上一节的代码,大体上按如下的思路: 功能拆分: 创建必要的工具类: 两个工具类StringUtils和TxtUtils. StringUtils,主要是获取当前系统的换行符: ...

  9. android传感器使用

    android传感器的使用以加速度传感器.方向传感器.磁场.压力.温度.光感.特别的距离传感器为例介绍. 首先所在的类需要implement SensorEventListener.使用传感器分为以下 ...

  10. HTTP和HTTPS的区别和联系

    超文本传输协议HTTP协议被用于在Web浏览器和网站服务器之间传递信息,HTTP协议以明文方式发送内容,不提供任何方式的数据加密,如果攻击者截取了Web浏览器和网站服务器之间的传输报文,就可以直接读懂 ...