两句闲话

  得到两个序列的最长公共子序列(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. DataNode启动不成功——java.net.BindException: Port in use: localhost:0 Caused by: java.net.BindException: Cannot assign requested address解决办法

    爱折腾的人总是会出线各种奇怪的问题.记得之前听一位大师讲过,我们不能踩完前进路上的所有坑前进,而应该学会怎样避开前进路上的坑,踩得坑越多,可能你的经验越丰富,但是付出的时间代价可能不是经验能换来的.我 ...

  2. hadoop21---使用代理修改List,代理流程

    package cn.itcast_05_proxy; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Met ...

  3. 利用同步网盘搭建个人或团队SVN服务器

    这篇文章是以前写的,现在强烈推荐两个站.1.http://git.oschina.com 2.http://www.coding.net. 推荐理由:1.可创建私有项目.2.免费稳定.3.VS2013 ...

  4. Dispose 与 close 方法 的区别

    Dispose : 释放 托管 与 非托管资源. Finalize : 释放 非托管资源. Close: 关闭资源后,可以再次使用资源.

  5. Sybase:删除表中的某列

    Sybases:删除表中的某列 alter table tb1(表名) drop clo1(列名); commit;

  6. CSS3圆盘时钟

    在线演示 本地下载

  7. java没有指针

    先说结论:java没有指针,它使用对象引用来替代指针 备注:c/c++的引用和java的引用完全不是一个东西 c/c++的引用是同一块内存的不同名字 java的引用指向一个对象,引用本身也占用了内存 ...

  8. 使用redis做mysql缓存

    应用Redis实现数据的读写,同时利用队列处理器定时将数据写入mysql. 同时要注意避免冲突,在redis启动时去mysql读取所有表键值存入redis中,往redis写数据时,对redis主键自增 ...

  9. 【P3355】骑士共存问题(最大流+黑白染色,洛谷)

    这个题刚看上去就让人不禁想到一道叫做方格取数问题的题目,事实上也就是这么做,对棋盘黑白染色,然后黑格子连源点,白的连汇点,点权为1.然后判断一下黑格子能影响到的白格子,边权为inf,跑一遍最大流就可以 ...

  10. 在NLP中深度学习模型何时需要树形结构?

    在NLP中深度学习模型何时需要树形结构? 前段时间阅读了Jiwei Li等人[1]在EMNLP2015上发表的论文<When Are Tree Structures Necessary for ...