全国咨询热线:18720358503

分销微信小程序_javascript完成最长公共子序列实例

类别:企业动态 发布时间:2021-01-08 浏览人次:

javascript实现最长公共子序列实例代码       mon sequence)mon substring)不是一回事儿,下面这篇文章主要给大家介绍了关于javascript实现最长公共子序列的相关资料,需要的朋友可以参考下。

介绍

最长公共子序列(Longest Common Subsequence LCS)是从给定的两个序列X和Y中取出尽可能多的一部分字符,按照它们在原序列排列的先后次序排列得到。LCS问题的算法用途广泛,如在软件不同版本的管理中,用LCS算法找到新旧版本的异同处;在软件测试中,用LCS算法对录制和回放的序列进行比较,在基因工程领域,用LCS算法检查患者DNA连与键康DNA链的异同;在防抄袭系统中,用LCS算法检查论文的抄袭率。LCS算法也可以用于程序代码相似度度量,人体运行的序列检索,视频段匹配等方面,所以对LCS算法进行研究具有很高的应用价值。

基本概念

子序列(subsequence): 一个特定序列的子序列就是将给定序列中零个或多个元素去掉后得到的结果(不改变元素间相对次序)。例如序列 A,B,C,B,D,A,B 的子序列有: A,B 、 B,C,A 、 A,B,C,D,A 等。

最长公共子序列:给定序列X和Y,从它们的所有公共子序列中选出长度最长的那一个或几个。
子串: 将一个序列从最前或最后或同时删掉零个或几个字符构成的新系列。区别与子序列,blogs这个字符串中子序列有多少个呢?很显然有27个,比如其中的cb,cgs等等都是其子序列

给一个图再解释一下:

我们可以看出子序列不见得一定是连续的,连续的是子串。

问题分析

我们还是从一个矩阵开始分析,自己推导出状态迁移方程。

首先,我们把问题转换成前端够为熟悉的概念,不要序列序列地叫了,可以认为是数组或字符串。一切从简,我们就估且认定是两个字符串做比较吧。

我们重点留意”子序列“的概念,它可以删掉多个或零个,也可以全部干掉。这时我们的第一个子序列为 空字符串 (如果我们的序列不是字符串,我们还可以 )!这个真是千万要注意到!许多人就是看不懂《算法导论》的那个图表,还有许多博客的作者不懂装懂。我们总是从左到右比较,当然了第一个字符串,由于作为矩阵的高,就垂直放置了。


假令X = "ABCDAB", Y="BDCABA",各自取出最短的序列,也就是空字符串与空字符串比较。LCS的方程解为一个数字,那么这个表格也只能填数字。两个空字符串的公同区域的长度为0.


然后我们X不动,继续让空字符串出阵,Y让“B”出阵,很显然,它们的公共区域的长度为0. Y换成其他字符, D啊,C啊, 或者, 它们的连续组合DC、 DDC, 情况没有变, 依然为0. 因此第一行都为0. 然后我们Y不动,Y只出空字任串,那么与上面的分析一样,都为0,第一列都是0.


LCS问题与背包问题有点不一样,背包问题还可以设置-1行,而最长公共子序列因为有空子序列的出现,一开始就把左边与上边固定死了。

然后我们再将问题放大些,这次双方都出一个字符,显然只有两都相同时,才有存在不为空字符串的公共子序列,长度也理解数然为1。

A为"X", Y为"BDCA"的子序列的任意一个


继续往右填空,该怎么填?显然,LCS不能大于X的长度,Y的从A字符串开始的子序列与B的A序列相比,怎么也能等于1。


如果X只从派出前面个字符A,B吧,亦即是“”,“A”, "B", "AB"这四种组合,前两个已经解说过了。那我们先看B,${X_1} == ${Y_0}, 我们得到一个新的公共子串了,应该加1。为什么呢?因为我们这个矩阵是一个状态表,从左到右,从上到下描述状态的迁移过程,并且这些状态是基于已有状态 累加 出来的。 现在我们需要确认的是,现在我们要填的这个格子的值与它周围已经填好的格子的值是存在何种关系 。目前,信息太少,就是一个孤点,直接填1。


然后我们让Y多出一个D做帮手,{"",A,B,AB} vs {"",B,D,BD},显然,继续填1. 一直填到Y的第二个B之前,都是1。 因为到BDCAB时,它们有另一个公共子序列,AB。


到这一步,我们可以总结一些规则了,之后就是通过计算验证我们的想法,加入新的规则或限定条件来完善。

Y将所有字符派上去,X依然是2个字符,经仔细观察,还是填2.

看五行,X再多派一个C,ABC的子序列集合比AB的子序列集合大一些,那么它与Y的B子序列集合大一些,就算不大,就不能比原来的小。显然新增的C不能成为战力,不是两者的公共字符,因此值应该等于AB的子序列集合。


并且我们可以确定,如果两个字符串要比较的字符不一样,那么要填的格子是与其左边或上边有关,那边大就取那个。

如果比较的字符一样呢,稍安毋躁,刚好X的C要与Y的C进行比较,即ABC的子序列集合{"",A,B,C,AB,BC,ABC}与BDC的子序列集合{"",B,D,C,BD,DC,BDC}比较,得到公共子串有“”,B,D 。这时还是与之前的结论一样,当字符相等时,它对应的格子值等于左边与右边与左上角的值,并且左边,上边,左上边总是相等的。这些奥秘需要更严格的数学知识来论证。

假设有两个数组,A和B。A[i]为A的第i个元素,A(i)为由A的第一个元素到第i个元素所组成的前缀。m(i, j)为A(i)和B(j)的最长公共子序列长度。

由于算法本身的递推性质,其实只要证明,对于某个i和j:

  m(i, j) = m(i-1, j-1) + 1 (当A[i] = B[j]时)

  m(i, j) = max( m(i-1, j), m(i, j-1) ) (当A[i] != B[j]时)

第一个式子很好证明,即当A[i] = B[j]时。可以用反证,假设m(i, j) m(i-1, j-1) + 1 (m(i, j)不可能小于m(i-1, j-1) + 1,原因很明显),那么可以推出m(i-1, j-1)不是最长的这一矛盾结果。

第二个有些trick。当A[i] != B[j]时,还是反证,假设m(i, j) max( m(i-1, j), m(i, j-1) )。

由反证假设,可得m(i, j) m(i-1, j)。这个可以推出A[i]一定在m(i, j)对应的LCS序列中(反证可得)。而由于A[i] != B[j],故B[j]一定不在m(i, j)对应的LCS序列中。所以可推出m(i, j) = m(i, j-1)。这就推出了与反正假设矛盾的结果。

得证。

我们现在用下面的方程来继续填表了。

程序实现

//by 司徒正美
function LCS(str1, str2){
 var rows = str1.split("")
 rows.unshift("")
 var cols = str2.split("")
 cols.unshift("")
 var m = rows.length 
 var n = cols.length 
 var dp = []
 for(var i = 0; i i++){ 
 dp[i] = []
 for(var j = 0; j j++){ 
 if(i === 0 || j === 0){
 dp[i][j] = 0
 continue
 if(rows[i] === cols[j]){ 
 dp[i][j] = dp[i-1][j-1] + 1 //对角+1
 }else{
 dp[i][j] = Math.max( dp[i-1][j], dp[i][j-1]) //对左边,上边取最大
 console.log(dp[i].join(""))//调试
 return dp[i-1][j-1]
 }

LCS可以进一步简化,只要通过挪位置,省去新数组的生成

//by司徒正美
function LCS(str1, str2){
 var m = str1.length 
 var n = str2.length
 var dp = [new Array(n+1).fill(0)] //第一行全是0
 for(var i = 1; i i++){ //一共有m+1行
 dp[i] = [0] //第一列全是0
 for(var j = 1; j j++){//一共有n+1列
 if(str1[i-1] === str2[j-1]){ 
 //注意这里,str1的第一个字符是在第二列中,因此要减1,str2同理
 dp[i][j] = dp[i-1][j-1] + 1 //对角+1
 } else {
 dp[i][j] = Math.max( dp[i-1][j], dp[i][j-1]) 
 return dp[m][n];
}

打印一个LCS

我们再给出打印函数,先看如何打印一个。我们从右下角开始寻找,一直找到最上一行终止。因此目标字符串的构建是倒序。为了避免使用stringBuffer这样麻烦的中间量,我们可以通过递归实现,每次执行程序时,只返回一个字符串,没有则返回一个空字符串, 以 printLCS(x,y,...) + str[i] 相加,就可以得到我们要求的字符串。

我们再写出一个方法,来验证我们得到的字符串是否真正的LCS字符串。作为一个已经工作的人,不能写的代码像在校生那样,不做单元测试就放到线上让别人踩坑。

//by 司徒正美,打印一个LCS
function printLCS(dp, str1, str2, i, j){
 if (i == 0 || j == 0){
 return "";
 if( str1[i-1] == str2[j-1] ){
 return printLCS(dp, str1, str2, i-1, j-1) + str1[i-1];
 }else{
 if (dp[i][j-1] dp[i-1][j]){
 return printLCS(dp, str1, str2, i, j-1);
 }else{
 return printLCS(dp, str1, str2, i-1, j);
//by司徒正美, 将目标字符串转换成正则,验证是否为之前两个字符串的LCS
function validateLCS(el, str1, str2){
 var re = new RegExp( el.split("").join(".*") )
 console.log(el, re.test(str1),re.test(str2))
 return re.test(str1) re.test(str2)
}

使用:

function LCS(str1, str2){
 var m = str1.length 
 var n = str2.length
 //....略,自行补充
 var s = printLCS(dp, str1, str2, m, n)
 validateLCS(s, str1, str2)
 return dp[m][n]
var c1 = LCS( "ABCBDAB","BDCABA");
console.log(c1) //4 BCBA、BCAB、BDAB
var c2 = LCS("" , "" );
console.log(c2) //5 34678 
var c3 = LCS("ACCGGTCGAGTGCGCGGAAGCCGGCCGAA" ,"GTCGTTCGGAATGCCGTTGCTCTGTAAA" );
console.log(c3) //20 GTCGTCGGAAGCCGGCCGAA


打印全部LCS

思路与上面差不多,我们注意一下,在LCS方法有一个Math.max取值,这其实是整合了三种情况,因此可以分叉出三个字符串。我们的方法将返回一个es6集合对象,方便自动去掉。然后每次都用新的集合合并旧的集合的字任串。

//by 司徒正美 打印所有LCS
function printAllLCS(dp, str1, str2, i, j){
 if (i == 0 || j == 0){
 return new Set([""])
 }else if(str1[i-1] == str2[j-1]){
 var newSet = new Set()
 printAllLCS(dp, str1, str2, i-1, j-1).forEach(function(el){
 newSet.add(el + str1[i-1])
 return newSet
 }else{
 var set = new Set()
 if (dp[i][j-1] = dp[i-1][j]){
 printAllLCS(dp, str1, str2, i, j-1).forEach(function(el){
 set.add(el)
 if (dp[i-1][j] = dp[i][j-1]){//必须用 =,不能简单一个else搞定
 printAllLCS(dp, str1, str2, i-1, j).forEach(function(el){
 set.add(el)
 return set
 }

使用:

function LCS(str1, str2){
 var m = str1.length 
 var n = str2.length
 //....略,自行补充
 var s = printAllLCS(dp, str1, str2, m, n)
 console.log(s)
 s.forEach(function(el){
 validateLCS(el,str1, str2)
 console.log("输出LCS",el)
 return dp[m][n]
var c1 = LCS( "ABCBDAB","BDCABA");
console.log(c1) //4 BCBA、BCAB、BDAB
var c2 = LCS("" , "" );
console.log(c2) //5 34678 
var c3 = LCS("ACCGGTCGAGTGCGCGGAAGCCGGCCGAA" ,"GTCGTTCGGAATGCCGTTGCTCTGTAAA" );
console.log(c3) //20 GTCGTCGGAAGCCGGCCGAA


空间优化

使用滚动数组:

function LCS(str1, str2){
 var m = str1.length 
 var n = str2.length
 var dp = [new Array(n+1).fill(0)],now = 1,row //第一行全是0
 for(var i = 1; i i++){ //一共有2行
 row = dp[now] = [0] //第一列全是0
 for(var j = 1; j j++){//一共有n+1列
 if(str1[i-1] === str2[j-1]){ 
 //注意这里,str1的第一个字符是在第二列中,因此要减1,str2同理
 dp[now][j] = dp[i-now][j-1] + 1 //对角+1
 } else {
 dp[now][j] = Math.max( dp[i-now][j], dp[now][j-1]) 
 now = 1- now; //1-1= 1-0= 1-1= 0 ...
 return row row[n]: 0
}

危险的递归解法

str1的一个子序列相应于下标序列{1, 2, …, m}的一个子序列,因此,str1共有${2^m}$个不同子序列(str2亦如此,如为${2^n}$),因此复杂度达到惊人的指数时间(${2^m * 2^n}$)。

//警告,字符串的长度一大就会爆栈
function LCS(str1, str2, a, b) {
 if(a === void 0){
 a = str1.length - 1
 if(b === void 0){
 b = str2.length - 1
 if(a == -1 || b == -1){
 return 0
 if(str1[a] == str2[b]) {
 return LCS(str1, str2, a-1, b-1)+1;
 if(str1[a] != str2[b]) {
 var x = LCS(str1, str2, a, b-1)
 var y = LCS(str1, str2, a-1, b)
 return x = y x : y
 }

参考链接
hrn1216/article/details/ a/41054 mysileng/archive/2013/05/

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对凡科的支持。


下一篇:没有了

推荐阅读

分销微信小程序_javascript完成最长公共子序列实例

javascript完成最多公共性子编码序列案例编码 mon sequence)mon substring)并不是一件事情儿,下边本文关键给大伙儿详细介绍了有关javascript完成最多公共性子编码序列的有关材料,必...

2021-01-08
展览会微信小程序_vue页面跳转后返回原页面初始

vue网页页面自动跳转后回到原网页页面原始部位方式 下边网编就为大伙儿共享一篇vue网页页面自动跳转后回到原网页页面原始部位方式,具备非常好的参照使用价值,期待对大伙...

2021-01-08
网站设计公司如何解决网站加载时间缓慢

    客户讨厌等候時间载入长的网页页面,网页页面速率再用户感受中起主要要功效,但很人忽视了速率提升,危害网站载入時间的重要要素有下列好多个。1、网络服务器网络服务器特...

2021-01-08
广州凡科互联网科技股份有限公司招聘天猫推广

招聘人数:7职位信息工作内容 :1、制定直通车/钻展推广/超级推荐投放计划并执行,根据推广数据实时优化调整,完成店铺销售目标,调控ROI;2、负责付费推广数据持续跟踪和评估,...

2021-01-08
微信预约小程序_vue 设置proxyTable参数进行代理跨

vue 设定proxyTable主要参数开展代理商跨域 _ 本文关键详细介绍了vue 设定proxyTable主要参数开展代理商跨域的有关材料,及代理商跨域的定义基本原理,必须的朋友能够参照下什么叫代...

2021-01-08
B2B2C多用户商城系统如何选择?

您的部位: > 新闻报道新闻资讯 > B2B2C要用户商城系统系统软件怎样挑选?近些年,伴随着互联网电子器件商务接待的迅速发展趋势,电子商务方式慢慢替代了传统式市场销售方式的影...

2021-01-08
X

400-8700-61718720358503
企业邮箱2639601583@qq.com
官方微信