两句闲话

  得到两个序列的最长公共子序列(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. cdoj1344卿学姐种美丽的花

    地址:http://acm.uestc.edu.cn/#/problem/show/1344 题目: 卿学姐种美丽的花 Time Limit: 8000/4000MS (Java/Others)    ...

  2. xshell如何同时打开多个标签

    查看标签>>>>>回话选项卡>>>>>> 打钩即可

  3. Apache 优化

    1.关闭DNS和名字解析 * HostnameLookups on | off | double 2.关闭客户端主机名解析 Apache .3之前HostnameLookups默认是打开的.这样客户端 ...

  4. JavaWeb HTML

    1. HTML介绍 1.1. 什么是HTML HTML的全称为Hyper Text Markup Language,译为超文本标记语言. 超文本,就是指页面内可以包含图片.链接,甚至音乐.程序等非文字 ...

  5. Linux之Shell 脚本加密工具-shc

    Much effort, much prosperity. 为什么要加密Shell脚本呢?当然是为了安全! 可能脚本里面涉及到密码之类的就需要进行加密了 一.下载安装shc工具 要保护自己编写的she ...

  6. Python 私有化属性

    # 对象的基本理论 # 什么事对象? # 万物皆对象 # 对象是具体物体 # 拥有属性 # 拥有行为 # 把很多零散的东西,封装成为一个整体 # 举例:王二小 # 属性 # 姓名 # 年龄 # 身高 ...

  7. 互联网开放平台API安全设计

    互联网开放平台设计1.需求:现在A公司与B公司进行合作,B公司需要调用A公司开放的外网接口获取数据,如何保证外网开放接口的安全性.2.常用解决办法:2.1 使用加签名方式,防止篡改数据2.2 使用Ht ...

  8. LeetCode 之 TwoSum

    题目: Given an array of integers, find two numbers such that they add up to a specific target number. ...

  9. Nginx的长链接

    网站使用程序discuz3访问都正常,只有用户登录存在异常,具体就是:用户登陆后会马上显示未登录,然后刷新一下又变成了登录中 这个问题的原因显然是由于session导致,后台有多个web机器,当用户登 ...

  10. [日常训练]Z国特色社会路

    Description 小$W$非常喜欢社会主义,这天他开始研究它的优越性. 他发现它们国家十分乐于修建特色的社会主义道路.具体的说,$Z$国有$n$座城市,由$m$条有向边连接,城市从$1$编号. ...