Poj 3356 ACGT(LCS 或 带备忘的递归)
题意:把一个字符串通过增、删、改三种操作变成另外一个字符串,求最少的操作数。
分析:
- 可以用LCS求出最大公共子序列,再把两个串中更长的那一串中不是公共子序列的部分删除。
- 分析可知两个字符串的距离肯定不会超过它们的长度之和,因为我们可以通过删除操作把两个串化为空串。如果两个字符串的第一个元素相同,则求A[2...ALen]和B[2...BLen]即可,如果不相同,则逐一分析增、删、改对下一步的影响:
- 删除A串的第一个字符,然后计算A[2...ALen]和B[1...BLen]即可。
- 删除B串的第一个字符,然后计算A[1...ALen]和B[2...BLen]即可。
- 修改A串的第一个字符,然后计算A[2...ALen]和B[2...BLen]即可。
- 修改B串的第一个字符,然后计算A[2...ALen]和B[2...BLen]即可。
- 增加A串的第一个字符到B串,然后计算A[2...ALen]和B[1...BLen]即可。
- 增加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 或 带备忘的递归)的更多相关文章
- POJ 3356 水LCS
题目链接: http://poj.org/problem?id=3356 AGTC Time Limit: 1000MS Memory Limit: 65536K Total Submission ...
- GIS部分理论知识备忘随笔
文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.高斯克吕格投影带换算 某坐标的经度为112度,其投影的6度带和3度带 ...
- Cheat (tldr, bropages) - Unix命令用法备忘单
cheat 是一个Unix命令行小工具,用来查询一些常用命令的惯用法(我们都知道,man page阅读起来太累了,常常是跳到最后去看 examples,但并不是所有man pages里面都有examp ...
- Nmap备忘单:从探索到漏洞利用(Part 5)
这是备忘单的最后一部分,在这里主要讲述漏洞评估和渗透测试. 数据库审计 列出数据库名称 nmap -sV --script=mysql-databases 192.168.195.130 上图并没有显 ...
- Nmap备忘单:从探索到漏洞利用(Part 4)
这是我们的Nmap备忘单的第四部分(Part 1. Part 2. Part 3).本文中我们将讨论更多东西关于扫描防火墙,IDS / IPS 逃逸,Web服务器渗透测试等.在此之前,我们应该了解一下 ...
- CSS系列:less备忘
less备忘 //这是一个运行在koala中的less文件,//注释不会被编译到css文件中,/**/注释会 ****************by 李可 2016/04/19 /*所有,所有伪类*/ ...
- 编写Windows Service 备忘
项目需求要做一个定时扫表,将按条件查询到的数据插入或者更新到另一个数据表的需求,老大要求让用window service来做 因为以前没有做过,把这次的经历写出来.作为备忘. 1.什么是windows ...
- 常用linux命令备忘
备忘: 关闭防火墙:# systemctl stop firewalld 查看防火墙状态:# systemctl status firewalld 停止防火墙:# systemctl disabl ...
- AngularJS之备忘与诀窍
译自:<angularjs> 备忘与诀窍 目前为止,之前的章节已经覆盖了Angular所有功能结构中的大多数,包括指令,服务,控制器,资源以及其它内容.但是我们知道有时候仅仅阅读是不够的. ...
随机推荐
- spring mvc注解和spring boot注解
1 spring mvc和spring boot之间的关系 spring boot包含spring mvc.所以,spring mvc的注解在spring boot总都是可以用的吗? spring b ...
- 京东android面试题(2018 顶级互联网公司面试题系列)
以下来自于北京的一个兄弟的面试题 1.静态内部类和非静态内部类有什么区别 2.谈谈你对java多态的理解 3.如何开启线程,run和runnable有什么区别 4.线程池的好处 5.说一下你知 ...
- ElasticSearch(二十)定位不合法的搜索及其原因
GET /test_index/test_type/_validate/query?explain { "query": { "math": { "t ...
- spring 配置c3p0连接池
一.导入与c3p0相关的jar包 二.xml配置文件 CombopooledDataSource类中提供了相应属性的set方法,因此可是使用属性注入的方式实例化对象. 三.示例 在userServic ...
- 存储过程 & 触发器
触发器主要是通过事件进行触发而被执行的,而存储过程可以通过存储过程名字而被直接调用.当对某一表进行诸如UPDATE. INSERT. DELETE 这些操作时, 就会自动执行触发器所定义的SQL 语句 ...
- mongodb 的注意点
昨天同事安装mongodb遇到了些问题,问了下我,后拉发现都是些细节没注意(讲道理这应该是很简单,一顿操作就ok的事情) 首先,下载 mongo包, 然后 ,解压安装, 启动之. 问题就出现在他后台启 ...
- centOS-64位通过YUM源安装nginx
第一步:在 /etc/yum.repos.d/ 目录下,建立名叫nginx.repo的软件源配置文件. 文件 nginx.repo 的内容是: [nginx] name=nginx re ...
- Intel Quick Sync Video Encoder
本篇记录Intel E3 1275处理器集成显卡的硬编码预研过程. 步骤如下: (1)环境搭建 (2)demo编译,测试 (3)研究demo源码,Media SDK API使用 (4)编写so动态库封 ...
- 阿里云CentOS7安装Docker
买了阿里云主机,由于学生有优惠,玩起来确实爽. 系统版本: [root@lxd ~]# cat /etc/redhat-release CentOS Linux release 7.0.1406 (C ...
- H5 input默认数字键盘,显示为密码格式
<P> <span class="yzname w25">银行密码</span> <input class="j_passwor ...