算法51:动态规划专练(力扣139题,单词拆分)---从左往右尝试模型的误区

本文主要是介绍算法51:动态规划专练(力扣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

这一题容易让人产生误导,一个target字符串,一个list数组,看起来非常像从左往右的尝试模型。一开始,我也是这么尝试的,按照从左往右尝试模型进行写递归版本代码的,写着写着发现不对劲。list数组中的单词可以重复使用,但是每个单词是固定的,不允许再拆分成逐个字母的。这一点和我之前写的贴纸拼词  算法34:贴纸拼词(力扣691题)-CSDN博客不一样

以下方为例:

输入: s = "catsandog", wordDict = ["cats", "dog", "san",  "cat"]  输出: true

如果以cat开头,那么第二个单词为 sand,第三个单词为dog,可以拼出来

如果以cats开头,那第二个单词就为and,第三个单词就为og了 拼不出来。

因此,需要对s进行讨论和判断了。

字符catsandog
下标012345678

1. 如果以c、ca为单词,list里面找不到

2.如果以cat为单词,list里面能够找到;这就是1中情况;那么下标为3的s就可能是第二个单词的开头

3.如果以cats为单词,list里面也能够找到,那么下标为4的a就可能是第二个单词的开头;

4. 如果第二个单词是以s开头的,san为单词,可以在list找到;此时,下标为6的d可能是下一个单词的开头了;

5.如果第二个单词是以a开头的,在list里面找不到;

6. 第三个单词,则是以d开头,dog在list里面可以找到;因此,可以拼出最终的s字符串;

代码如何实现呢?

package code04.动态规划专项训练03;import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;/*** 力扣 139题 : 单词拆分* https://leetcode.cn/problems/word-break/description/?envType=study-plan-v2&envId=dynamic-programming** 思路:* 1. 统计list中存在的无重复单词* 2. 字符串从0处开始往后找*     a. 如果字符串中能否切割出一个完整的单词,这个单词是在list表中出现的。那么就把这个单词末尾字符的下一个下标标记为新单词的开始位置,即dp表对应下标标记为true*     b. 如果找到末尾都找不到,就一直找。直到本轮递归结束* 3. 找到 dp 表中 对应为 true的下标。 从此下标开始可以组成一个新单词。具体逻辑与步骤2相同.*    举例说明:*    s = "catsandog"*    wordDict = ["cats", "dog", "sand", "and", "cat"]**    字符串     c a t s a n d d o g*    dp表下标   0 1 2 3 4 5 6 7 8 9*    dp表下标   T F F T T F F T F F*    下标0处 为 true 一个字符串如果开头位置*    下标3处 为 true 因为cat在字典中能够找到,所以下标3可能是一个新单词的开头*    下标4处 为 true 因为cats在字典中能够找到,所以下标4可能是一个新单词的开头*    下标7处 为 true 无论是sand 还是 and 都是单词。因此他们末尾的下一个下标,即index=7可能是一个新单词的开头** 4. 如果整个字符串都可以由字典list中单词拼接出来。那么这个字符串一定可以走到末尾。而dp表的长度是比字符串的长多多1. 即dp*    表的末尾肯定为true。*/
public class WordBreak_官方题解_02 {public boolean wordBreak(String s, List<String> wordDict){//统计list中无重复单词Set<String> wordDictSet = new HashSet(wordDict);//记录哪些位置可以是单词的开始位置boolean[] dp = new boolean[s.length() + 1];//默认从下标为0处为单词的开始位置dp[0] = true;for (int i = 1; i <= s.length(); i++) {for (int j = 0; j < i; j++) {//dp[j] = true, 代表j这个位置是一个单词的开头if (dp[j] && wordDictSet.contains(s.substring(j, i))) {//i代表本来循环的末尾位置。因此 String.substring(j, i) 是 [j,i)//i的前一个位置为单词的末尾位置。那么i就是新单词的开始位置或者空的占位dp[i] = true;break;}}}return dp[s.length()];}public static void main(String[] args) {WordBreak_官方题解_02 s = new WordBreak_官方题解_02();String a = "leetcode";List<String> list2 = new ArrayList<>();list2.add("leet");list2.add("code");System.out.println(s.wordBreak(a, list2));/* List<String> list = new ArrayList<>();list.add("cats");list.add("dog");list.add("sand");list.add("and");list.add("cat");String s1 = "catsandog";System.out.println(s.wordBreak(s1, list));*/}
}

这一题倒不难,只是由于我之前习惯性的按照动态规划的固定模板化思路去解题,容易进入误区;

老是想着套路、模型去解题,有时候确实事半功倍;但是,这一题却反其道而行,让你看着像是之前的解题套路,结果就迷茫了。

下一题,我将分享另一个看着像从左往右模型套路的题。但是,由于数据量比较大,肯定要排除掉从左往右模型的题目。

这篇关于算法51:动态规划专练(力扣139题,单词拆分)---从左往右尝试模型的误区的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/833124

相关文章

基于Python实现一个图片拆分工具

《基于Python实现一个图片拆分工具》这篇文章主要为大家详细介绍了如何基于Python实现一个图片拆分工具,可以根据需要的行数和列数进行拆分,感兴趣的小伙伴可以跟随小编一起学习一下... 简单介绍先自己选择输入的图片,默认是输出到项目文件夹中,可以自己选择其他的文件夹,选择需要拆分的行数和列数,可以通过

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

springboot如何通过http动态操作xxl-job任务

《springboot如何通过http动态操作xxl-job任务》:本文主要介绍springboot如何通过http动态操作xxl-job任务的问题,具有很好的参考价值,希望对大家有所帮助,如有错... 目录springboot通过http动态操作xxl-job任务一、maven依赖二、配置文件三、xxl-

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

MyBatis编写嵌套子查询的动态SQL实践详解

《MyBatis编写嵌套子查询的动态SQL实践详解》在Java生态中,MyBatis作为一款优秀的ORM框架,广泛应用于数据库操作,本文将深入探讨如何在MyBatis中编写嵌套子查询的动态SQL,并结... 目录一、Myhttp://www.chinasem.cnBATis动态SQL的核心优势1. 灵活性与可

详解如何使用Python从零开始构建文本统计模型

《详解如何使用Python从零开始构建文本统计模型》在自然语言处理领域,词汇表构建是文本预处理的关键环节,本文通过Python代码实践,演示如何从原始文本中提取多尺度特征,并通过动态调整机制构建更精确... 目录一、项目背景与核心思想二、核心代码解析1. 数据加载与预处理2. 多尺度字符统计3. 统计结果可

Mybatis嵌套子查询动态SQL编写实践

《Mybatis嵌套子查询动态SQL编写实践》:本文主要介绍Mybatis嵌套子查询动态SQL编写方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、实体类1、主类2、子类二、Mapper三、XML四、详解总结前言MyBATis的xml文件编写动态SQL

SpringBoot整合Sa-Token实现RBAC权限模型的过程解析

《SpringBoot整合Sa-Token实现RBAC权限模型的过程解析》:本文主要介绍SpringBoot整合Sa-Token实现RBAC权限模型的过程解析,本文给大家介绍的非常详细,对大家的学... 目录前言一、基础概念1.1 RBAC模型核心概念1.2 Sa-Token核心功能1.3 环境准备二、表结

SpringBoot实现Kafka动态反序列化的完整代码

《SpringBoot实现Kafka动态反序列化的完整代码》在分布式系统中,Kafka作为高吞吐量的消息队列,常常需要处理来自不同主题(Topic)的异构数据,不同的业务场景可能要求对同一消费者组内的... 目录引言一、问题背景1.1 动态反序列化的需求1.2 常见问题二、动态反序列化的核心方案2.1 ht