单词拆分

作者:xiao_ben_zhu
链接:https://leetcode-cn.com/problems/word-break/solution/shou-hui-tu-jie-san-chong-fang-fa-dfs-bfs-dong-tai/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

给定一个非空字符串 s 和一个包含非空单词的列表 wordDict,判定 s 是否可以被空格拆分为一个或多个在字典中出现的单词。

说明:

拆分时可以重复使用字典中的单词。
你可以假设字典中没有重复的单词。
示例 1:

输入: s = “leetcode”, wordDict = [“leet”, “code”]
输出: true
解释: 返回 true 因为 “leetcode” 可以被拆分成 “leet code”。
示例 2:

输入: s = “applepenapple”, wordDict = [“apple”, “pen”]
输出: true
解释: 返回 true 因为 “applepenapple” 可以被拆分成 “apple pen apple”。
注意你可以重复使用字典中的单词。
示例 3:

输入: s = “catsandog”, wordDict = [“cats”, “dog”, “sand”, “and”, “cat”]
输出: false

力扣原题

问题分析:
DFS 思路
"leetcode"能否break,可以拆分为:"l"是否是单词表的单词、剩余子串能否break,"le"是否是单词表的单词、剩余子串能否break……
用 DFS 回溯,考察所有的拆分可能,指针从左往右扫描 s 串:
1. 如果指针的左侧部分是单词表中的单词,则对右侧的剩余子串,递归考察。
2. 如果指针的左侧部分不是单词表的单词,不用看了,回溯,考察别的分支。
在这里插入图片描述

    public boolean wordBreak(String s, List<String> wordDict) {
        return dfs(s, 0, wordDict);
    }

    public boolean dfs(String s, int start, List<String> wordDict){
        if(start == s.length()){
            return true;
        }
        
        for(int i=start;i<s.length();i++){
            if(wordDict.indexOf(s.substring(start, i+1))!=-1 && dfs(s, i+1, wordDict)){
                return true;
            }
        }
        return false;
    }

该方法会大量重复计算:
在这里插入图片描述
加入记忆化
我们用一个数组,存计算的结果,数组索引为指针位置,值为计算的结果。下次遇到相同的子问题,直接返回数组中的缓存值。

    public boolean wordBreak(String s, List<String> wordDict) {
        Boolean[] memory = new Boolean[s.length()];
        return dfs(s, 0, wordDict, memory);
    }

    public boolean dfs(String s, int start, List<String> wordDict, Boolean[] memory){
        if(start == s.length()){
            return true;
        }
        if(memory[start]!=null)return memory[start];

        for(int i=start;i<s.length();i++){
            if(wordDict.indexOf(s.substring(start, i+1))!=-1 && dfs(s, i+1, wordDict, memory)){
                memory[start] = true;
                return true;
            }
        }
        memory[start] = false;
        return false;
    }

动态规划

  • s 串能否分解为单词表的单词,即:前 s.length 个字符的 s 串能否分解为单词表单词。
  • 将大问题分解为规模小一点的子问题,规模就是这个长度:
  • 前 i 个字符的子串能否分解成单词表单词 + 剩余子串是否为单个单词。

状态转移方程
在这里插入图片描述

  • [0, i][0,i] 区间子串 的 dp[i+1]dp[i+1] 是否为真,取决于两部分:
  • 它的前缀子串 [0, j-1][0,j−1] 的 dp[j]dp[j] ,是否为真
    剩余子串 [j,i][j,i] ,是否是一个单词表单词
    base case
    dp[0] = true。长度为 0 的子串是由单词表的单词组成的。是很诡异。
    但,这只是为了让边界情况也能满足状态转移方程,即上图:当黄色部分为空字符串时,dp[i+1]dp[i+1] 全然取决于 [0,i][0,i] 子串是否为一个单词表单词。
    所以我们让 dp[0] = true
public boolean wordBreak(String s, List<String>wordDict){
        boolean[] dp = new boolean[s.length()+1];
        dp[0] = true;
        for(int i=1;i<=s.length();i++){
            for(int j=0;j<i;j++){
                String sub = s.substring(j, i);
                if(dp[j]&&wordDict.contains(sub)){
                   dp[i] = true;
                   break;
                }
            }
        }
        return dp[s.length()];
    }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值