Lintcode--005

Given two strings, find the longest common subsequence (LCS).     最长公共子序列

Your code should return the length of LCS.

Clarification

What‘s the definition of Longest Common Subsequence?

Example

For "ABCD" and "EDCA", the LCS is "A" (or "D", "C"), return 1.

For "ABCD" and "EACB", the LCS is "AC", return 2.

应用: 在生物工程中,比较两个DNA串的相似性。

解题:

1. 首先,必须弄清楚什么是子序列,什么是最长公共子序列;

2. 容易想的方法就是暴力求解,穷举所有子序列,然后找到最大的,然而这种指数级别的复杂度肯定不合适。

3. 所以我们需要分析问题,刻画公共子序列所具有的特征;

假设: 有两个字符串 A = “ABCDUHNEK” ;  B = ”KFACEKLO“  ;求解他两的最大公共子序列;

首先:两个字符串的长度设为: m=A.length(), n=B.length();

我们考虑两个字符串的最后一个字符A[m-1] 和 B[n-1] 。

(1)假如相等:两个字符串的最长公共子序列就一定包含最后一个字符,而且它的长度应该等价于字符串 A‘ = “ABCDUHNE” 与 B’ = ”KFACEKL“ 的最大公共子序列的长度+1

(2)假如不相等:两个字符串的最长公共子序列就有两种情况:等价于,A = “ABCDUHNEK” 与  B‘ = ”KFACEKL“ 的最大公子序列长度 或者

A‘ = “ABCDUHNE” 与  B = ”KFACEKLO“ 的最大公共子序列长度。

有此特点可以得到这个问题的递归解公式:

其中c[i,j]定义为一个二维数组,用于存储两个字符串最长公共子序列的长度。下标i,j表示字符串A中的前i-1个字符和 字符串B中的前 j-1个字符所具有的最长公共子串(注意:数组计数从0开始)。

递归求解,可得类似下表:

X字符串:ABCBDAB

Y字符串:BDCABA

算法时间复杂度为:Θ(m + n)。

实现代码如下:

class Solution {
public:
    /**
     * @param A, B: Two strings.
     * @return: The length of longest common subsequence of A and B.
     */
    int longestCommonSubsequence(string A, string B) {
        // write your code here

        //最大公共子序列,方法在算法导论中有专门讲解。当学完之后来看,题目很简单。
        //编程习惯要好!  注意括号对称,循环嵌套缩进!!!
        int a=A.size();//两个字符串的长度
        int b=B.size();
        int num; //最大公共子序列长度
        int c[a+1][b+1];//定义二维数组,用来存放最大公共子序列的长度,注意定义和引用时下标的区别;
        for(int i=0;i<a+1;i++){
            c[i][0]=0;
        }
        for(int j=1;j<b+1;j++){
            c[0][j]=0;
        }

        for(int i=0;i<a;i++){
            for(int j=0;j<b;j++){
                if(A[i]==B[j]){
                    c[i+1][j+1]=c[i][j]+1;
                }
                else{
                    if(c[i][j+1]>=c[i+1][j]){
                        c[i+1][j+1]=c[i][j+1];
                    }
                    else{
                        c[i+1][j+1]=c[i+1][j];
                    }
                }
            }
        }
        num=c[a][b];
        return num;
    }
};

附:当需要返回最长公共子序列时,只需要在上面所考虑的三中情况下都设置相应的标记位就能实现了。表格中所画的三种箭头,在实际实现时可以用0,1,-1三个标志位来表示。

时间: 2024-12-16 20:47:55

Lintcode--005的相关文章

[lintcode the-smallest-difference]最小差(python)

题目链接:http://www.lintcode.com/zh-cn/problem/the-smallest-difference/ 给定两个整数数组(第一个是数组 A,第二个是数组 B),在数组 A 中取 A[i],数组 B 中取 B[j],A[i] 和 B[j]两者的差越小越好(|A[i] - B[j]|).返回最小差. 排好序后用两个指针分别扫描两个数组,每次更新他们的差值的绝对值.并且依据他们两个数字的大小来决定谁来移动指针. 1 class Solution: 2 # @param

lintcode.44 最小子数组

最小子数组 描述 笔记 数据 评测 给定一个整数数组,找到一个具有最小和的子数组.返回其最小和. 注意事项 子数组最少包含一个数字 您在真实的面试中是否遇到过这个题? Yes 哪家公司问你的这个题? Airbnb Amazon LinkedIn Cryptic Studios Dropbox Apple Epic Systems TinyCo Yelp Hedvig Zenefits Uber Snapchat Yahoo Microsoft Bloomberg Facebook Google

windows_learn 005 软件部署

windows_learn 005  软件部署 内容总览 软件部署概述 将软件发布给用户 将软件分配给用户或计算机 将软件升级和重新部署 部署Microsoft Office 发布ZAP应用程序 软件部署的其它设置 将软件重新封装成MSI应用程序 第5章 使用组策略部署软件(p207) 可以通过AD DS组策略来为企业内部计算机部署(Deploy)软件,也就是自动为这些计算机 安装.维护和删除软件. 软件部署概述 即通过组策略来将软件部署给域内的计算机,来实现自动化安装和部署软件给客户端,而 软

lintcode 66.67.68 二叉树遍历(前序、中序、后序)

AC代码: /** * Definition of TreeNode: * public class TreeNode { * public int val; * public TreeNode left, right; * public TreeNode(int val) { * this.val = val; * this.left = this.right = null; * } * } */ public class Solution { /** * @param root: The r

[LintCode/LeetCode]——两数和、三数和、四数和

LintCode有大部分题目来自LeetCode,但LeetCode比较卡,下面以LintCode为平台,简单介绍我AC的几个题目,并由此引出一些算法基础. 1)两数之和(two-sum) 题目编号:56,链接:http://www.lintcode.com/zh-cn/problem/two-sum/ 题目描述: 给一个整数数组,找到两个数使得他们的和等于一个给定的数 target. 你需要实现的函数twoSum需要返回这两个数的下标, 并且第一个下标小于第二个下标.注意这里下标的范围是 1

Lintcode 469. 等价二叉树

----------------------------------------------- AC代码: /** * Definition of TreeNode: * public class TreeNode { * public int val; * public TreeNode left, right; * public TreeNode(int val) { * this.val = val; * this.left = this.right = null; * } * } */

Lintcode 75.寻找峰值

--------------------------------------- 按照给定的峰值定义,峰值的左半部分一定是递增的,所以只要找到不递增的即可. AC代码: class Solution { /** * @param A: An integers array. * @return: return any of peek positions. */ public int findPeak(int[] A) { for(int i=1;i<A.length;i++){ if(A[i]>=

Lintcode 9.Fizz Buzz 问题

------------------------ AC代码: class Solution { /** * param n: As description. * return: A list of strings. */ public ArrayList<String> fizzBuzz(int n) { ArrayList<String> results = new ArrayList<String>(); for (int i = 1; i <= n; i++

Lintcode 97.二叉树的最大深度

--------------------------------- AC代码: /** * Definition of TreeNode: * public class TreeNode { * public int val; * public TreeNode left, right; * public TreeNode(int val) { * this.val = val; * this.left = this.right = null; * } * } */ public class S

[Lintcode two-sum]两数之和(python,双指针)

题目链接:http://www.lintcode.com/zh-cn/problem/two-sum/ 给一个整数数组,找到两个数使得他们的和等于一个给定的数target. 备份一份,然后排序.搞两个指针分别从左从右开始扫描,每次判断这两个数相加是不是符合题意,如果小了,那就把左边的指针向右移,同理右指针.然后在备份的数组里找到位置. 1 class Solution: 2 """ 3 @param numbers : An array of Integer 4 @param