LeetCode-热题100:139. 单词拆分

题目描述

给你一个字符串 s 和一个字符串列表 wordDict 作为字典。如果可以利用字典中出现的一个或多个单词拼接出 s 则返回 true

**注意:**不要求字典中出现的单词全部都使用,并且字典中的单词可以重复使用。

示例 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

提示:

  • 1 <= s.length <= 300
  • 1 <= wordDict.length <= 1000
  • 1 <= wordDict[i].length <= 20
  • s 和 wordDict[i] 仅由小写英文字母组成
  • wordDict 中的所有字符串互不相同

代码及注释

go 复制代码
func wordBreak(s string, wordDict []string) bool {
    // 初始化一个 map,用于存储 wordDict 中的单词
    wordDictSet := make(map[string]bool)
    
    // 初始化动态规划数组,dp[i] 表示 s 的前 i 个字符是否可以被 wordDict 中的单词拆分
    dp := make([]bool, len(s) + 1)
    
    // 将 wordDict 中的单词加入到 wordDictSet 中
    for _, w := range wordDict {
        wordDictSet[w] = true
    }
    
    // 初始化,空字符串可以被拆分
    dp[0] = true
    
    // 遍历 s 的每个字符
    for i := 1; i <= len(s); i++ {
        // 遍历 j,0 ≤ j < i
        for j := 0; j < i; j++ {
            // 如果 s 的前 j 个字符可以被拆分,并且 s[j:i] 存在于 wordDict 中
            if dp[j] && wordDictSet[s[j:i]] {
                // 则 s 的前 i 个字符可以被拆分
                dp[i] = true
                break
            }
        }
    }
    
    // 返回 s 是否可以被完全拆分
    return dp[len(s)]
}

代码解释

  1. 初始化 wordDictSet

    go 复制代码
    wordDictSet := make(map[string]bool)
    • 创建一个 map,用于存储 wordDict 中的单词,以便于后续快速查找。
  2. 初始化动态规划数组 dp

    go 复制代码
    dp := make([]bool, len(s) + 1)
    • 创建一个布尔类型的数组 dp,其中 dp[i] 表示字符串 s 的前 i 个字符是否可以被 wordDict 中的单词拆分。
  3. 将 wordDict 中的单词加入到 wordDictSet 中

    go 复制代码
    for _, w := range wordDict {
        wordDictSet[w] = true
    }
    • 遍历 wordDict,将其中的单词加入到 wordDictSet 中,以便后续快速查找。
  4. 初始化 dp[0] 为 true

    go 复制代码
    dp[0] = true
    • 空字符串可以被拆分,因此 dp[0] 初始化为 true
  5. 动态规划遍历

    go 复制代码
    for i := 1; i <= len(s); i++ {
        for j := 0; j < i; j++ {
            if dp[j] && wordDictSet[s[j:i]] {
                dp[i] = true
                break
            }
        }
    }
    • 使用动态规划的方法遍历字符串 s 的每个字符,对于每个字符 i,从 0i-1 的字符 j 进行遍历。
    • 如果 dp[j]true(即字符串 s 的前 j 个字符可以被拆分),并且 s[j:i] 存在于 wordDict 中,那么 dp[i] 就设置为 true
    • 使用 break 结束内层循环,因为只需要找到一个满足条件的 j 即可。
  6. 返回结果

    go 复制代码
    return dp[len(s)]
    • 返回 dp[len(s)],表示整个字符串 s 是否可以被完全拆分为 wordDict 中的单词。
相关推荐
萧萧玉树1 小时前
B树系列详解
数据结构·b树
XuanRanDev5 小时前
【数据结构】树的基本:结点、度、高度与计算
数据结构
王老师青少年编程5 小时前
gesp(C++五级)(14)洛谷:B4071:[GESP202412 五级] 武器强化
开发语言·c++·算法·gesp·csp·信奥赛
DogDaoDao5 小时前
leetcode 面试经典 150 题:有效的括号
c++·算法·leetcode·面试··stack·有效的括号
Coovally AI模型快速验证6 小时前
MMYOLO:打破单一模式限制,多模态目标检测的革命性突破!
人工智能·算法·yolo·目标检测·机器学习·计算机视觉·目标跟踪
可为测控7 小时前
图像处理基础(4):高斯滤波器详解
人工智能·算法·计算机视觉
Milk夜雨7 小时前
头歌实训作业 算法设计与分析-贪心算法(第3关:活动安排问题)
算法·贪心算法
Ai 编码助手7 小时前
在 Go 语言中如何高效地处理集合
开发语言·后端·golang
BoBoo文睡不醒7 小时前
动态规划(DP)(细致讲解+例题分析)
算法·动态规划
apz_end8 小时前
埃氏算法C++实现: 快速输出质数( 素数 )
开发语言·c++·算法·埃氏算法