题意:把一个字符串通过增、删、改三种操作变成另外一个字符串,求最少的操作数。

分析:

  • 可以用LCS求出最大公共子序列,再把两个串中更长的那一串中不是公共子序列的部分删除。
  • 分析可知两个字符串的距离肯定不会超过它们的长度之和,因为我们可以通过删除操作把两个串化为空串。如果两个字符串的第一个元素相同,则求A[2...ALen]和B[2...BLen]即可,如果不相同,则逐一分析增、删、改对下一步的影响:
  1. 删除A串的第一个字符,然后计算A[2...ALen]和B[1...BLen]即可。
  2. 删除B串的第一个字符,然后计算A[1...ALen]和B[2...BLen]即可。
  3. 修改A串的第一个字符,然后计算A[2...ALen]和B[2...BLen]即可。
  4. 修改B串的第一个字符,然后计算A[2...ALen]和B[2...BLen]即可。
  5. 增加A串的第一个字符到B串,然后计算A[2...ALen]和B[1...BLen]即可。
  6. 增加B串的第一个字符到A串,然后计算A[1...ALen]和B[2...BLen]即可。

总之,一步操作以后,会出现三种情况,A[2...ALen]和B[1...BLen],A[1...ALen]和B[2...BLen],A[2...ALen]和B[2...BLen],这样可以用递归求解了。但是递归求解时有些数据会被重复计算,所以使用一个二维数组来记录已经计算过的情况。

其实,这两个解法的本质是相同的,不过分析的角度不一样。递归可以改成非递归的形式,会和LCS的dp解法大同小异。

LCS:

package LCS;

import java.util.Scanner;

public class Poj_ACGT {
static int[][] dp=new int[1001][1001];
public static void main(String[] args) {
// TODO Auto-generated method stub
Scanner sc=new Scanner(System.in);
while(sc.hasNext()){
for(int i=0;i<1001;i++){
for(int j=0;j<1001;j++){
dp[i][j]=0;
}
}
int sLen=sc.nextInt();
String s=sc.next();
int tLen=sc.nextInt();
String t=sc.next(); for(int i=1;i<=sLen;i++){
for(int j=1;j<=tLen;j++){
if(s.charAt(i-1)==t.charAt(j-1))
dp[i][j]=dp[i-1][j-1]+1;
else
dp[i][j]=Math.max(dp[i-1][j],dp[i][j-1]);
}
}
System.out.println(Math.max(sLen, tLen)-dp[sLen][tLen]);
}
}
}

带备忘的递归:

package Recursive;

import java.util.Scanner;

/**
* Memory:9136K Time:219MS
*/
public class Poj_AGTC {
static int[][] dp=new int[1001][1001];
public static int count(String s,int sB,int sE,String t,int tB,int tE){
if(sB > sE){
if(tB>tE){
return 0;
}else{
return tE-tB+1;
}
}
if(tB>tE){
if(sB > sE){
return 0;
}else{
return sE-sB+1;
}
}
if(s.charAt(sB) == t.charAt(tB)){
if(dp[sB+1][tB+1] == 0){
dp[sB+1][tB+1]=count(s,sB+1,sE,t,tB+1,tE);
}
return dp[sB+1][tB+1];
}else{
if(dp[sB][tB+1] == 0){
dp[sB][tB+1]=count(s,sB,sE,t,tB+1,tE);
}
if(dp[sB+1][tB] == 0){
dp[sB+1][tB]=count(s,sB+1,sE,t,tB,tE);
}
if(dp[sB+1][tB+1] == 0){
dp[sB+1][tB+1]=count(s,sB+1,sE,t,tB+1,tE);
}
int min=Math.min(dp[sB][tB+1], dp[sB+1][tB]);
min=Math.min(min, dp[sB+1][tB+1]);
return min+1;
}
}
public static void main(String[] args) {
// TODO Auto-generated method stub
Scanner sc=new Scanner(System.in);
while(sc.hasNext()){
for(int i=0;i<1001;i++){
for(int j=0;j<1001;j++){
dp[i][j]=0;
}
}
int sLen=sc.nextInt();
String s=sc.next();
int tLen=sc.nextInt();
String t=sc.next();
System.out.println(count(s,0,sLen-1,t,0,tLen-1));
}
}
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

Poj 3356 ACGT(LCS 或 带备忘的递归)的更多相关文章

  1. POJ 3356 水LCS

    题目链接: http://poj.org/problem?id=3356 AGTC Time Limit: 1000MS   Memory Limit: 65536K Total Submission ...

  2. GIS部分理论知识备忘随笔

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.高斯克吕格投影带换算 某坐标的经度为112度,其投影的6度带和3度带 ...

  3. Cheat (tldr, bropages) - Unix命令用法备忘单

    cheat 是一个Unix命令行小工具,用来查询一些常用命令的惯用法(我们都知道,man page阅读起来太累了,常常是跳到最后去看 examples,但并不是所有man pages里面都有examp ...

  4. Nmap备忘单:从探索到漏洞利用(Part 5)

    这是备忘单的最后一部分,在这里主要讲述漏洞评估和渗透测试. 数据库审计 列出数据库名称 nmap -sV --script=mysql-databases 192.168.195.130 上图并没有显 ...

  5. Nmap备忘单:从探索到漏洞利用(Part 4)

    这是我们的Nmap备忘单的第四部分(Part 1. Part 2. Part 3).本文中我们将讨论更多东西关于扫描防火墙,IDS / IPS 逃逸,Web服务器渗透测试等.在此之前,我们应该了解一下 ...

  6. CSS系列:less备忘

    less备忘 //这是一个运行在koala中的less文件,//注释不会被编译到css文件中,/**/注释会 ****************by 李可 2016/04/19 /*所有,所有伪类*/ ...

  7. 编写Windows Service 备忘

    项目需求要做一个定时扫表,将按条件查询到的数据插入或者更新到另一个数据表的需求,老大要求让用window service来做 因为以前没有做过,把这次的经历写出来.作为备忘. 1.什么是windows ...

  8. 常用linux命令备忘

    备忘: 关闭防火墙:# systemctl stop firewalld 查看防火墙状态:#  systemctl status firewalld 停止防火墙:#  systemctl disabl ...

  9. AngularJS之备忘与诀窍

    译自:<angularjs> 备忘与诀窍 目前为止,之前的章节已经覆盖了Angular所有功能结构中的大多数,包括指令,服务,控制器,资源以及其它内容.但是我们知道有时候仅仅阅读是不够的. ...

随机推荐

  1. 【python】-- 类的继承(新式类/经典类)、多态

    继承 之前我们说到了类的公有属性和类的私有属性,其实就是类的封装,现在准备随笔的 是继承,是面向对象的第二大特性. 面向对象编程 (OOP) 语言的一个主要功能就是“继承”.继承是指这样一种能力:它可 ...

  2. Django之权限用法

    **记住每一个url都是一个权限** 注册 可插拔试的权限,可以先写其他的逻辑,在最后再把权限加上 将rbac组件拷贝到项目上,注册项目 修改表结构 将写好的用户表对rbac的User表进行一对一的关 ...

  3. hdoj 1455 Sticks 【dfs】

    题意:找最短的木棍可以组成的长度, hdoj  1518 的加强版 代码: #include <stdio.h> #include <string.h> #include &l ...

  4. html post

    post请求对应的html页面 页面效果 html代码 <html> <body> <form method="post" > First na ...

  5. 在vi或vim上查找字符串

    从开头搜索 在命令模式下,输入/你要查找的字符 按下回车,可以看到vim把光标移动到该字符处 再按n(小写)查看下一个匹配 按N(大写)查看上一个匹配, capslock切换大小写,也可以在小写状态下 ...

  6. vim 正则表达式查找ip

    特别说明: \v \v 表示 very magic 这种模式下很多字符默认就表示一些特殊意义,而不用加 \ 如 : < 单词开头 > 单词结尾 ( 组开始 ) 组结束 { 次数开始 } 次 ...

  7. 每天一个Linux命令(28)df命令

    报告文件系统磁盘空间的使用情况.获取硬盘被占用了多少空间,目前还剩下多少空间等信息.       (1)用法:       用法:  df [选项] [文件]       (2)功能: 功能:  显示 ...

  8. Data Structure Array: Maximum circular subarray sum

    http://www.geeksforgeeks.org/maximum-contiguous-circular-sum/ #include <iostream> #include < ...

  9. [原创]java WEB学习笔记07:关于HTTP协议

    本博客为原创:综合 尚硅谷(http://www.atguigu.com)的系统教程(深表感谢)和 网络上的现有资源(博客,文档,图书等),资源的出处我会标明 本博客的目的:①总结自己的学习过程,相当 ...

  10. Redis的管理

    一.redis持久化 redis是内存数据库,一切的数据都是存储到内存中的,我们知道,当服务器意外关机,那么在内存中的数据都将丢失,但是redis为我们提供持久化功能,这样就能把数据保存到硬盘上.re ...