LCS引论

在这篇博文中,博主要给大家讲一个算法----最长公共子序列(LCS)算法。我最初接触这个算法是在高中学信息学竞赛的时候。那时候花了好长时间理解这个算法。老师经常说,这种算法是母算法,即从这种算法中能衍生出许许多多的子算法。是的,博主在大二的算法导论考试中,就被LCS衍生的子算法“坑”了。当时一道动态规划题目,大概是要写一列书中放置书架的问题。博主联想到了LCS,但是最后还是只是用暴力求解法解出了那道题目,甚是遗憾。在这里,我也对此算法做一个详解。

优化子结构

首先,我们来看优化子结构:

设X=(x 1 , ..., x m ) 、Y=(y 1 , ..., y n )  是两个序列,Z=(z 1 , ..., z k ) 是X 与Y 的LCS ,我们有:

⑴     如果xm =yn,  则zk =xm = yn , Zk-1 是Xm-1 和Yn-1 的LCS , 即 :

LCS XY  = LCS X m-1 Y n-1 + <xm =yn >.

⑵     如果xm ≠ yn , 且zk≠ xm , 则Z 是X m-1 和Y 的

LCS , 即  LCS XY = LCS X m-1 Y

⑶     如果xm ≠ yn ,且zk≠ yn ,则Z 是X 与Y n-1 的LCS ,即 :

LCS XY = LCS XY n-1

构造C[m,n]数组表示xm 、yn 的LCS长度,因此,求得X和Y的优化解结构的递归方程为:

C[i, j] = 0 if i=0  或 j=0

C[i, j] = C[i-1, j-1] + 1 if i, j>0  且 x i  = y j

C[i, j] = Max(C[i, j-1], C[i-1, j]) if i, j>0  且 x i ≠ y j

在存储LCS的时候,我们根据已经构造好的B[m,n]来寻找全部的LCS。

如果B[m,n]=0,说明X和Y当前比对的字符相同,因此我们把X和Y都向前一个字符,再进行比对。

如果B[m,n]=1,说明xm ≠ yn , 且zk≠ xm ,此时我们只用把X向前移一位,再进行比对。

如果B[m,n]=3,说明xm ≠ yn ,且zk≠ yn,此时我们需要把Y向前移动一位,再进行比对。

如果B[m,n]=2,说明c[i - 1][j] = c[i][j - 1],我们需要将两个字符串依次向前移动一位,进行比对

关键的数据结构及简单说明

C[m,n]: C[i,j] 是X i 与Y j 的LCS 的长度;

B[m,n]: B[i,j] 是指针 ,指向计算C[i,j] 时所选择的子问题的优化解所对应的C 表的表项。通俗的说,B[m,n]记录的是轨迹;

public static String[] Log:存储全部LCS字符串;

public static char[] lcs:字符数组,存储

TreeSet<String> tree = new TreeSet<String>():将LCS数组存入Tree集合中,供打印使用。

示例程序

 package test;

import java.util.Scanner;

import java.util.TreeSet;

public class LCS {

public static StringBuffer X;

public static StringBuffer Y;

public static int m;

public static int n;

public static int len;

public static int[][] b;

public static int[][] c;

public static String[] Log;

public static char[] lcs;

public final static int MAX = 100;

public static int boardlen;

public static void main(String[] args) {

Log = new String[MAX];

LCS lcs = new LCS();

Scanner in = new Scanner(System.in);

System.out.println("string X:");

X = new StringBuffer(in.next());

System.out.println("string Y:");

Y = new StringBuffer(in.next());

lcs_length();

System.out.println("LCS:");

store_lcs(m, n, len);

PrintLCS();

X.setLength(0);

Y.setLength(0);

in.close();

}

public static void lcs_length() {

m = X.length();

n = Y.length();

lcs = new char[m + 1];

boardlen = 0;

c = new int[m + 1][n + 1];

b = new int[m + 1][n + 1];

for (int i = 1; i <= m; i++)

c[i][0] = 0;

for (int j = 0; j <= n; j++)

c[0][j] = 0;

for (int i = 1; i <= m; i++){

for (int j = 1; j <= n; j++) {

if (X.charAt(i - 1) == Y.charAt(j - 1)) {

c[i][j] = c[i - 1][j - 1] + 1;

b[i][j] = 0;

} else if (c[i - 1][j] > c[i][j - 1]) {

c[i][j] = c[i - 1][j];

b[i][j] = 1;

} else if (c[i - 1][j] == c[i][j - 1]) {

c[i][j] = c[i - 1][j];

b[i][j] = 2;

} else {

c[i][j] = c[i][j - 1];

b[i][j] = 3;

}

}

}

len = c[m][n];

}

public static void store_lcs(int m, int n, int Len) {

if (m == 0 || n == 0) {

Log[boardlen] = new String(lcs);

boardlen++;

} else {

if (b[m][n] == 0) {

lcs[Len] = X.charAt(m - 1);

Len--;

store_lcs(m - 1, n - 1, Len);

} else if (b[m][n] == 3) {

store_lcs(m, n - 1, Len);

} else if (b[m][n] == 1) {

store_lcs(m - 1, n, Len);

} else {

store_lcs(m, n - 1, Len);

store_lcs(m - 1, n, Len);

}

}

}

public static void PrintLCS() {

TreeSet<String> tree = new TreeSet<String>();

for (int i = 0; i <boardlen; i++) {

tree.add(Log[i]);

}

String[] string = new String[tree.size()];

for (int i = 0; i < string.length; i++) {

string[i] = tree.pollFirst();

System.out.println(string[i]);

}

}

public void printit() {

for (int i = 0; i < Log.length; i++) {

if (Log[i] != null) {

System.out.println(Log[i]);

}

}

}

}

后序

程序经本人测试,能够运行出正确结果。以上仅供大家学习交流,转载请注明出处。

我的第一篇博客----LCS学习笔记的更多相关文章

  1. 第一篇博客:Hello World

    2016年10月10日,双十,好日子,决定开始写第一篇博客,标题想了会,就叫Hello World 吧,哈哈^_^. 首先感谢博客园的管理们能批准我的申请,记得在14年的时候申请过一次,竟然没申请通过 ...

  2. 我的第一篇博客 ——【ToDoList】小程序开发

    我是一只即将大四的大三狗,这是我的第一篇博客,说来惭愧.今年1月份,学校放寒假的时候开始自学的IOS,放假的时候比较起劲,看了一堆Object-C的视频,然后照着中英文对照的IOS基础开发教程,做了两 ...

  3. Hello World -- 第一篇博客

    今年注定是不寻常的一年,因为技术,接触了许多大牛.通过一篇篇博文,看到了大牛们勤奋好学.孜孜不倦的精神,于是决定也开个博客,向大牛学习. 博客开了,写点什么呢?奈何肚子里墨水不多,吐出来也多是白沫,不 ...

  4. “Hello, my first blog”------第一篇博客的仪式感

    本人在校大学生一枚,开通博客,主要是想记录自己的学习过程,分享自己的学习经历.记得大一的时候,很多不懂的操作和知识,都是在博客上找到了相应的解决办法.但比较讽刺的是,很多时候,曾经解决了的问题,当再次 ...

  5. C博客作业00—我的第一篇博客

    C博客作业00-我的第一篇博客 1. 你对网络专业或者计算机专业了解是怎样? 泛泛了解 - 原先只知道网络工程隶属于计算机工程学院,与院中其他专业一样,同样都需要学习大量的计算机基础知识,然后再分支学 ...

  6. Hello World -- 第一篇博客 -- 活着的意义

    今年注定是不寻常的一年,因为技术,接触了许多大牛.通过一篇篇博文,看到了大牛们勤奋好学.孜孜不倦的精神,于是决定也开个博客,向大牛学习. 博客开了,写点什么呢?奈何肚子里墨水不多,吐出来也多是白沫,不 ...

  7. Youcans 的第一篇博客

    这是我的第一篇博客. 今后我会将我的学习心得和总结在这里发布,与大家共享,共勉.

  8. 我的第一篇博客blog,笑哭

    我的第一篇博客blog Markdown学习 一级标题:#加一个空格 加 文字, 二级标题:加2个##以此类推 字体 粗体:hello world!字体前有二个星号,字体后有二个星号 斜体:hello ...

  9. 第一篇博客:HTML:background的使用

    开篇 我是一名程序员小白,这是我写的第一篇博客,在学习的路上难免会遇到难以解决的问题,我将会在这里写下我遇到的问题并附上解决方法 希望可以对各位有所帮助!! 我们在html中经常会遇到这样的问题 例如 ...

随机推荐

  1. 用Maven部署war包到远程Tomcat服务器

    过去我们发布一个Java Web程序通常的做法就是把它打成一个war包,然后用SSH这样的工具把它上传到服务器,并放到相应的目录里,让Tomcat自动去解包,完成部署. 很显然,这样做不够方便,且我们 ...

  2. handlebars自定义helper的写法

    handlebars相对来讲算一个轻量级.高性能的模板引擎,因其简单.直观.不污染HTML的特性,我个人特别喜欢.另一方面,handlebars作为一个logicless的模板,不支持特别复杂的表达式 ...

  3. 到爱尔兰敲代码 / Come, Coding in Ireland

    这是我在都柏林的第四个月,该办的证也都办完了,该安定下来的也安定下来了,所以也简单介绍下到爱尔兰做IT的相关过程和政策. 如果有兴趣在英语环境工作的话,我也可以帮忙推荐或者找找. 去年15年1月正好开 ...

  4. JVM系列-常用参数

    1.堆内存 堆内存用于存储new对象,垃圾回收器负责堆内存的管理.但Java程序实际占用的空间则由堆内存.栈内存(程序运行栈).程序计数器.常量区.代码区.本地内存等. 堆内存分为Young和Old, ...

  5. 学会使用Spring注解

      概述 注释配置相对于 XML 配置具有很多的优势: 它可以充分利用 Java 的反射机制获取类结构信息,这些信息可以有效减少配置的工作.如使用 JPA 注释配置 ORM 映射时,我们就不需要指定 ...

  6. 安卓Design包之TabLayout控件的简单使用

    Google在2015的IO大会上,给我们带来了更加详细的Material Design设计规范,同时,也给我们带来了全新的Android Design Support Library,在这个supp ...

  7. Android指纹识别深入浅出分析到实战(6.0以下系统适配方案)

    指纹识别这个名词听起来并不陌生,但是实际开发过程中用得并不多.Google从Android6.0(api23)开始才提供标准指纹识别支持,并对外提供指纹识别相关的接口.本文除了能适配6.0及以上系统, ...

  8. 嵌入式服务器jetty,让你更快开发web

    概述 jetty是什么? jetty是轻量级的web服务器和servlet引擎. 它的最大特点是:可以很方便的作为嵌入式服务器. 它是eclipse的一个开源项目.不用怀疑,就是你常用的那个eclip ...

  9. Java进击C#——语法之多线程

    本章简言 上一章中笔者对C#一些独有的语法点进行讲解,相信也可以看C#的一些神奇之处.那么本章主要是放在多线程这方面的知识.不管是C#还是JAVA在开发过程或多或少都会用到关于多线程的编程.当然笔者不 ...

  10. C#中HttpClient使用注意:预热与长连接

    最近在测试一个第三方API,准备集成在我们的网站应用中.API的调用使用的是.NET中的HttpClient,由于这个API会在关键业务中用到,对调用API的整体响应速度有严格要求,所以对HttpCl ...