码迷,mamicode.com
首页 > 其他好文 > 详细

动态规划1.求最长回文子串

时间:2018-12-28 00:45:21      阅读:320      评论:0      收藏:0      [点我收藏+]

标签:problem   har   end   推导   过程   body   优化   longest   amp   

求字符串的子串大致有四中方法,暴力,DP,中心拓展,马拉车算法,这篇讲DP怎么做。

DP最重要的就是要能利用到前面的结果来推断当前状态,比暴力优化的地方就在此,暴力需要对每一个字符串做一次O(n)的操作才能判断出结果,也就是整个过程要O(n^3),但DP对每一个字符串的判断时间是O(1),总共是O(n^2)

假设s=adcdf,我们可以的推导过程就像下图(i代表行,表示结尾,表示字符串结尾,j代表列,表示字符串开头)

1 2 3 4 5
1 a
2 ad d
3 adc dc c
4 adcd dcd cd d
5 adcdf dcdf cdf df f

我们用dp[][]来表示所有可能子串是否回文,比如dp[1][2]表示第ad,不回文,所以dp[1][2]=false,根据上面表格可以总结一下
1.当i=j时,dp[j][i]=true
2.i-j=1时,比如dp[1][2]为ad,表示两个相邻的字符,此时我们只要判断str[1]==str[2]就能得出dp[1][2]的结果
3.i-j>1时,我们来看dp[j]ij],首先还是要判断开头和结尾是否相等,也就是判断
str[i]==str[j],假如此时str[i]=str[j],我们还要再看剩下的子串是否回文,
我们可以直接从dp[j+1][i-1]来判断剩下的子串,把结果直接拿来用

dp[j][i]=(str[i]==str[j]) ;i-j<=1
dp[j][i]=(str[i]==str[j])&&dp[j+1][i-1];i-j>1

class Solution {
    public String longestPalindrome(String s) {
        if(s.isEmpty()==true) return "";
        int Len=s.length();
        if(Len==1) return s;
        Boolean[][] dp=new Boolean[Len][Len];
        int len=0,max=0,start=0,end=0;      
        for(int i=0;i<Len;i++) {
            for(int j=0;j<=i;j++) {
                if(i-j>1) {
                    dp[j][i]=((s.charAt(i)==s.charAt(j))&&dp[j+1][i-1]);
                }else {
                    dp[j][i]=(s.charAt(i)==s.charAt(j));
                }
                len=i-j;
                if(dp[j][i]==true&&len>=max) { 
                    max=len;
                    start=j;
                    end=i;
                } 
            }
        }
        if(start==end) {
            String str="";
            return str+s.charAt(0);
        }           
        return s.substring(start, end+1);
    }
}

题目链接:点这里

动态规划1.求最长回文子串

标签:problem   har   end   推导   过程   body   优化   longest   amp   

原文地址:https://www.cnblogs.com/jchen104/p/10188284.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!