力扣解题思路:分割回文串

131. 分割回文串

思路:
在这里插入图片描述
因为题目要求不是分割的方法种数,而是具体的分割方法,所以用dfs比较合适,那如果直接用dfs暴力遍历各种可能显然是不高效的,因此我们可以用一个set记录所有出现的回文子串(记忆化递归),这就不用每遇到一个子串就判断一次是不是回文字串了:

class Solution {
    List<List<String>> res = new ArrayList<>();
    Set<String> set = new HashSet<>();
    public List<List<String>> partition(String s) {
        if(s.length() == 0) return res;

        dfs(s,0,new ArrayList<>());

        return res;  
    }

    public void dfs(String s,int start,List<String> list){
        if(start == s.length()){
            res.add(new ArrayList<>(list));
            return;
        }
        for(int i=start;i<s.length();i++){
            if(set.contains(s.substring(start,i+1)) || is(s.substring(start,i+1))){
                list.add(s.substring(start,i+1));
                dfs(s,i+1,list);
                list.remove(list.size()-1);
            }
        }
    }

    public boolean is(String sub){
        if(sub.length() == 0) return false;
        int i = 0;
        int j = sub.length()-1;
        while(i<j){
            if(sub.charAt(i) != sub.charAt(j)){
                return false;
            }
            i++;
            j--;
        }
        set.add(sub);
        return true;
    }
}

132. 分割回文串 II

思路:
在这里插入图片描述

    public int minCut(String s) {
        int len = s.length();
        if (len < 2) {
            return 0;
        }

        int[] dp = new int[len];
        for (int i = 0; i < len; i++) {
            dp[i] = i;
        }

        for (int i = 1; i < len; i++) {
            if (checkPalindrome(s, 0, i)) {
                dp[i] = 0;
                continue;
            }
            // 当 j == i 成立的时候,s[i] 就一个字符,一定是回文
            // 因此,枚举到 i - 1 即可
            for (int j = 0; j < i; j++) {
                if (checkPalindrome(s, j + 1, i)) {
                    dp[i] = Math.min(dp[i], dp[j] + 1);
                }
            }
        }
        return dp[len - 1];
    }

    private boolean checkPalindrome(String s, int left, int right) {
        while (left < right) {
            if (s.charAt(left) != s.charAt(right)) {
                return false;
            }
            left++;
            right--;
        }
        return true;
    }