zoukankan      html  css  js  c++  java
  • 【力扣】5. 最长回文子串

    给你一个字符串 s,找到 s 中最长的回文子串。

    示例 1:

    输入:s = "babad"
    输出:"bab"
    解释:"aba" 同样是符合题意的答案。
    示例 2:

    输入:s = "cbbd"
    输出:"bb"
    示例 3:

    输入:s = "a"
    输出:"a"
    示例 4:

    输入:s = "ac"
    输出:"a"
     

    提示:

    1 <= s.length <= 1000
    s 仅由数字和英文字母(大写和/或小写)组成


    来源:力扣(LeetCode)
    链接:https://leetcode-cn.com/problems/longest-palindromic-substring
    著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

    //暴力破解
            //时间复杂度 O(n^3)
            //空间复杂度 O(1)
        //那就找到所有的子串,挨个遍历,直到找到最长的子串
        public String longestPalindrome(String s) {
    
            //若是长度,不足,则返回原字符串
            if(s.length() <= 1){
                return s;
            }
    
            int start = 0,maxLen = 1;
    
            char[] charArray = s.toCharArray();
    
            int length = s.length();
    
            for(int i = 0; i < length -1 ; i++){
    
                for( int j = i+1; j< length ;j++){
                    if(j - i + 1 > maxLen && isPalindrome(charArray,i,j)){
                        maxLen = j - i + 1;
                        start = i;
                    }
                }
            }
    
            return s.substring(start,start + maxLen);
            
        }
    
    
    
        /**
        * 判断是否为回文子串
        **/
        public boolean isPalindrome(char[] charArray , int start, int end){
            while(start < end){
                if(charArray[start] != charArray[end]){
                    return false;
                }
                start++;
                end--;
            }
            return true;
        }
    //动态规划
        //时间复杂度O(n^2)
        //空间复杂度O(n^2)
        public String longestPalindrome(String s) {
            //若是长度,不足,则返回原字符串
            if(s.length() <= 1){
                return s;
            }
    
    
            //动态规划最重要的就是找到公式
                //
    
            int start = 0, maxLen = 1; //定义最大长度,以及当前最大长度的初始位置
    
            char[] charArray = s.toCharArray();
    
            int length = s.length();
    
            // dp[i][j] 表示 s[i, j] 是否是回文串
            boolean[][] dp = new boolean[length][length];
    
            //初始化开始值,这个二维数组的斜线为true
            for(int i = 0; i < length ; i++){
                dp[i][i] = true;
            }
    
            //你可能回疑问,为什么要这么循环
                //例子:
                    //假如我们要求值:F(1,4) 的是否为回文串,而且 arr[1] = arr[4] ,那么我们应该想要知道 F(2,3)是不是回文串,那么就要在f(1,4)之前求出来f(2,3)的值。
            for(int j = 1; j < length ; j++){
                for( int i = j-1; i>=0 ;i--){
                    //假设当前字符串的left != right
                    if(charArray[i] != charArray[j]){
                        dp[i][j] = false;
                    } else {
                        //若是当前字符串的left == right
                        if(j - i < 3){
                            //当前字符串长度为3个,直接设置为true
                            dp[i][j] = true;
                        } else {
                            //需要看下里面的子串是否为回文串
                            dp[i][j] = dp[i+1][j-1];
                        }
                        if(dp[i][j] && j - i + 1 > maxLen){
                            start = i;
                            maxLen = j-i+1;
                        }
                    }
                }
            }
            return s.substring(start,start + maxLen);
        }

    中心扩散:

    public String longestPalindrome(String s) {
            int len = s.length();
            if (len < 2) {
                return s;
            }
            int maxLen = 1;
            String res = s.substring(0, 1);
            // 中心位置枚举到 len - 2 即可
            for (int i = 0; i < len - 1; i++) {
                String oddStr = centerSpread(s, i, i);
                String evenStr = centerSpread(s, i, i + 1);
                String maxLenStr = oddStr.length() > evenStr.length() ? oddStr : evenStr;
                if (maxLenStr.length() > maxLen) {
                    maxLen = maxLenStr.length();
                    res = maxLenStr;
                }
            }
            return res;
        }
    
        private String centerSpread(String s, int left, int right) {
            // left = right 的时候,此时回文中心是一个字符,回文串的长度是奇数
            // right = left + 1 的时候,此时回文中心是一个空隙,回文串的长度是偶数
            int len = s.length();
            int i = left;
            int j = right;
            while (i >= 0 && j < len) {
                if (s.charAt(i) == s.charAt(j)) {
                    i--;
                    j++;
                } else {
                    break;
                }
            }
            // 这里要小心,跳出 while 循环时,恰好满足 s.charAt(i) != s.charAt(j),因此不能取 i,不能取 j
            return s.substring(i + 1, j);
        }

    在这里要注意一个细节:回文串在长度为奇数和偶数的时候,“回文中心”的形式是不一样的。

    奇数回文串的“中心”是一个具体的字符,例如:回文串 "aba" 的中心是字符 "b";
    偶数回文串的“中心”是位于中间的两个字符的“空隙”,例如:回文串串 "abba" 的中心是两个 "b" 中间的那个“空隙”。

    引用:

    https://leetcode-cn.com/problems/longest-palindromic-substring/solution/zhong-xin-kuo-san-dong-tai-gui-hua-by-liweiwei1419/

    一个入行不久的Java开发,越学习越感觉知识太多,自身了解太少,只能不断追寻
  • 相关阅读:
    Apollo的Oracle适配改动
    尝试Java,从入门到Kotlin(下)
    尝试Java,从入门到Kotlin(上)
    RabbitMQ权限控制原理
    一文彻底掌握二叉查找树(多组动图)(史上最全总结)
    图解:深度优先搜索与广度优先搜索及其六大应用
    图解:如何理解与实现散列表
    图解:什么是“图”?
    查找算法系列文(一)一文入门二叉树
    线性表(数组、链表、队列、栈)详细总结
  • 原文地址:https://www.cnblogs.com/fengtingxin/p/14364542.html
Copyright © 2011-2022 走看看