从使用回溯分割字符串的技巧到前向搜索

题目 131. 分割回文串

给你一个字符串 s,请你将 s 分割成一些子串,使每个子串都是 回文串 。返回 s 所有可能的分割方案。

回文串 是正着读和反着读都一样的字符串。

答案:

class Solution {boolean[][] f;List<List<String>> ret = new ArrayList<List<String>>();List<String> ans = new ArrayList<String>();int n;public List<List<String>> partition(String s) {n = s.length();f = new boolean[n][n];for (int i = 0; i < n; ++i) {Arrays.fill(f[i], true);}for (int i = n - 1; i >= 0; --i) {for (int j = i + 1; j < n; ++j) {f[i][j] = (s.charAt(i) == s.charAt(j)) && f[i + 1][j - 1];}}dfs(s, 0);return ret;}public void dfs(String s, int i) {if (i == n) {ret.add(new ArrayList<String>(ans));return;}for (int j = i; j < n; ++j) {if (f[i][j]) {ans.add(s.substring(i, j + 1)); dfs(s, j + 1); // #A 为什么这里入参是j+1而不是i+1ans.remove(ans.size() - 1);}}}
}作者:力扣官方题解
链接:https://leetcode.cn/problems/palindrome-partitioning/solutions/639633/fen-ge-hui-wen-chuan-by-leetcode-solutio-6jkv/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

疑问:dfs(s, j + 1); 入口 为什么这里入参是j+1而不是i+1?

首先看一下dfs(s, j + 1);入口参数是j+1的回溯过程:

以aab为例,我们dfs的流程是:
添加"a"--->
继续添加"a"--->
继续添加"b"--->
继续dfs发现i==len越界,将答案["a","a","b"]加入到ret里然后返回-->
在res中删除最后添加的"b",并且发现当前层dfs的for循环不能再执行了于是自然返回--->
res继续删除末尾的"a",再次for循环更长的回文串,但发现"ab"不是回文串,而且for循环执行不下去了遂返回--->
res继续删除末尾的"a"(空了),然后for循环(i,j)(0,0)变为(0,1),将对应回文串"aa"添加--->
继续添加"b"-->继续dfs发现i==len越界,将答案["aa", "b"]加入到ret然后返回--->
在res中删除最后添加的"b",发现for不能执行了自然返回--->
res继续删除"aa",再次for循环找更长的回文串,(i,j)(0,1)变成了(0,2),但"子串aab"不是回文串---->
继续for循环,发现j==len越界,for循环结束,最外层调用的dfs函数自然返回。主函数返回结果集ret...

打印路径如下:

(7个节点)
i:0,j:0,s.substring(i,j+1):a
i:1,j:1,s.substring(i,j+1):a
i:2,j:2,s.substring(i,j+1):b
i:1,j:2,s.substring(i,j+1):ab
i:0,j:1,s.substring(i,j+1):aa
i:2,j:2,s.substring(i,j+1):b
i:0,j:2,s.substring(i,j+1):aab

再看一下下面这个棵回溯树:

root
|_ 'a' (0,0)|_ 'a' (1,1)|_ 'b' (2,2)|_ 'ab' (1,2)
|_ 'aa' (0,1)|_ 'b' (2,2)
|_ 'aab' (0,2)

把它画成图:
在这里插入图片描述

使用j+1的树中,为什么没有|_ ‘aab’ (0,2)的子树

在使用 j+1 的方法中,代码逻辑会优先尝试将字符串划分为更小的部分,也就是说,它会先尝试将 aab 分割为 a 和 ab,再分割为 a、a 和 b。因此,在这个过程中,它实际上并没有尝试到 aab 这一整个字符串,也就没有 ‘aab’ (0,2) 这个节点。这是因为,一旦发现一个回文串,它就会立即向后查找,而不会尝试更大的字符串。根本原因是 if (f[i][j])的判断

dfs(s, j + 1);入口参数是i+1的回溯历程

同样的字符串“aab”,其打印路径如下:

(共经过9个节点)
i:0,j:0,s.substring(i,j+1):a
i:1,j:1,s.substring(i,j+1):a
i:2,j:2,s.substring(i,j+1):b
i:1,j:2,s.substring(i,j+1):ab
i:0,j:1,s.substring(i,j+1):aa
i:1,j:1,s.substring(i,j+1):a
i:2,j:2,s.substring(i,j+1):b
i:1,j:2,s.substring(i,j+1):ab
i:0,j:2,s.substring(i,j+1):aab

对应的树如下:
在这里插入图片描述

使用i+1的树中,为什么没有|_ ‘aab’ (0,2)的子树?

对于 aab,‘aab’ (0,2) 是整个字符串,是无法继续划分的,所以没有子树。之所以在使用 i+1 的树中有 aab 这一节点,是因为每次 dfs 的入参增加了1,所以即使前一步已经找到了一个完整的回文串(如 aa),也会尝试下一步,从而导致 aab 被添加到树中。但是因为 aab 不是回文串,所以不会有对应的有效划分,即没有子树。

区别

可以看出,改变 dfs(s, j + 1) 为 dfs(s, i + 1)后,树中多出了一些节点,这是为什么呢?使用j+1作为入参后,似乎更像是做切割,表明之前的部分与此无关,后者更像是还会重复做一些判断

使用 dfs(s, j + 1)dfs(s, i + 1) 的差别主要体现在搜索的方向和重复判断上。

  1. dfs(s, j + 1): 这是一种前向搜索的策略,每次找到一个回文串,都会立即查找该回文串后面的所有可能划分。这意味着,当你进行下一次递归调用时,你已经完成了当前位置之前的所有搜索,你将不需要再次考虑这些部分。也就是说,这种策略有效地避免了重复判断。

  2. dfs(s, i + 1): 这是一种深度搜索策略,每次只前进一个位置,然后尝试所有可能的划分。这意味着,在每个位置,你都需要考虑所有的可能划分,即使这些划分在之前已经被考虑过。因此,这种策略会产生一些重复的判断和多余的节点。

这就是为什么改变 dfs(s, j + 1)dfs(s, i + 1) 后,树中会多出一些节点。而使用 j + 1 的策略可以减少这种重复,使得搜索更高效。

什么是前向搜索

前向搜索(forward search)是一种搜索策略,在这种策略下,我们从一个起始状态开始,然后逐步探索所有可达的状态。这种策略常常在图论、树、动态规划等问题中使用。在你的代码中,前向搜索就体现在 dfs(s, j + 1) 这一行,即每次找到一个回文串后,立即查找该回文串之后的所有可能划分。

前向搜索是一种剪枝策略嘛

前向搜索本身并不是一种剪枝策略,而是一种搜索策略。它决定了搜索的方向和步骤,即从当前状态出发,探索所有可以直接到达的状态。然而,在某些情况下,前向搜索可以减少搜索的空间,提高搜索的效率,这可以看作是一种隐式的剪枝。比如在这个问题中,dfs(s, j+1) 就相当于一种前向搜索策略,它避免了不必要的重复搜索,减小了搜索空间

前向搜索相关力扣题目

在力扣(LeetCode)上,许多题目都可以使用前向搜索策略,例如:

题目22. 括号生成

题目39. 组合总和

class Solution {List<List<Integer>>res;List<Integer>tmp;public List<List<Integer>> subsets(int[] nums) {int n=nums.length;res=new ArrayList<>();tmp=new ArrayList<>();dfs(nums,0);return res;}void dfs(int[] nums,int i){res.add(new ArrayList<>(tmp));for(int j=i;j<nums.length;j++){tmp.add(nums[j]);dfs(nums,j+1);tmp.remove(tmp.size()-1);}}
}

题目46. 全排列
题目79. 单词搜索
这些题目都可以通过前向搜索的方式,从一个状态开始,逐步探索所有可能的状态。

如果使用i+1搜索,应该是全量搜索对吧,列出一些相关力扣题目

是的,使用 i+1 的方法相当于全量搜索。它会尝试所有的可能划分,包括一些在实际中并不需要的划分。在力扣(LeetCode)上,很多需要遍历所有可能解的题目都可以使用这种方法,例如:

题目78. 子集
题目90. 子集 II
题目216. 组合总和 III
题目377. 组合总和 Ⅳ

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/15060.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【多线程中的线程安全问题】线程互斥

1 &#x1f351;线程间的互斥相关背景概念&#x1f351; 先来看看一些基本概念&#xff1a; 1️⃣临界资源&#xff1a;多线程执行流共享的资源就叫做临界资源。2️⃣临界区&#xff1a;每个线程内部&#xff0c;访问临界资源的代码&#xff0c;就叫做临界区。3️⃣互斥&…

【密码学】三、AES

AES 1、AES产生2、数学基础2.1有限域GF(2^8^)2.1.1加法运算2.1.2乘法运算2.1.3x乘运算2.1.4系数在GF(2^8^)上的多项式 3、AES算法描述3.1字节代换3.2行移位3.3列混合3.4轮密钥加3.5密钥扩展 1、AES产生 征集AES算法的活动&#xff0c;目的是确定一个非保密的、公开的、全球免费…

HCIP——重发布及路由策略实验

重发布及路由策略实验 一、实验拓扑二、实验要求三、实验思路三、实验步骤1、配置接口IP地址以及环回地址2、配置动态路由协议3、重发布4、更改接口类型5、配置路由策略 一、实验拓扑 二、实验要求 1、使用双点双向重发布2、所有路由器进行最佳选路3、存在备份路径&#xff0c…

软考05根据内存区域大小计算芯片数量

文章目录 前言一、原题二、解题思路1.计算内存区域的大小2.计算每个存储器芯片的容量3.计算芯片数量 总结 前言 从网上看题答案是有了&#xff0c;但是不知道具体的计算过程就很难受&#xff0c;不然下次还是不会&#xff0c;只能自己梳理了 一、原题 二、解题思路 1.计算内存…

Android开发之Fragment动态添加与管理

文章目录 主界面布局资源两个工具Fragment主程序 主界面布局资源 在activity_main.xml中&#xff0c;声明两个按钮备用&#xff0c;再加入一个帧布局&#xff0c;待会儿用来展示Fragment。 <?xml version"1.0" encoding"utf-8"?> <LinearLayo…

手机的python怎么运行文件,python在手机上怎么运行

大家好&#xff0c;小编来为大家解答以下问题&#xff0c;手机上的python怎么运行程序&#xff0c;手机的python怎么运行文件&#xff0c;今天让我们一起来看看吧&#xff01; 1、python程序怎么在手机上运行 python语言应用很广泛&#xff0c;自己也很喜欢使用它&#xff0c;其…

iOS - 检测项目中无用类和无用图片

一、无引用图片检测 LSUnusedResources 安装插件 LSUnusedResources &#xff0c;用【My Mac】模拟器运行,如下图&#xff1a; Project Path 就是项目所在的路径&#xff0c;然后点击右下角 Search按钮&#xff0c;就可以看到被搜索出来的图片资源。 注意&#xff1a;这里被搜…

Linux——进程控制

目录 1. 进程创建 1.1 fork函数 1.2 fork系统调用内部宏观流程 1.3 fork后子进程执行位置分析 1.4 fork后共享代码分析 1.5 fork返回值 1.6 写时拷贝 1.7 fork常规用法 1.8 fork调用失败的原因 2.进程终止 2.1 进程退出场景 2.2 strerror函数—返回描述错误号的字符…

解决问题:python PermissionError: [WinError 5]拒绝访问

重要&#xff1a;关闭PyCharm Community Edition 2022.3等与python相关的编程程序找到按照python解释器的位置python->右键>属性>安全->点击组或用户名"中的Users->编辑点击"组或用户名"中的Users->把"完全控制"打钩->应用->…

Servlet文件的下载

第一种方法直接在前端使用超链接&#xff0c;也就是a标签 浏览器不能识别会直接下载&#xff08;像压缩文件不能直接下载&#xff09;&#xff0c;浏览器能识别&#xff0c;想要下载加一个download属性。download可以不写任何信息。 首先在web下建一个文件&#xff0c;放需要…

在Windows 10和11中恢复已删除的照片

可以在Windows 10或11上恢复已删除的照片吗&#xff1f; 随着技术的发展&#xff0c;越来越多的用户习惯在电子设备上存储照片。如果这些照片被删除&#xff0c;可能会给用户带来重大损失。当照片丢失时&#xff0c;您可能会想是否可以恢复已删除的照片&#xff1f; …

Kafka原理剖析

一、简介 Kafka是一个分布式的、分区的、多副本的消息发布-订阅系统&#xff0c;它提供了类似于JMS的特性&#xff0c;但在设计上完全不同&#xff0c;它具有消息持久化、高吞吐、分布式、多客户端支持、实时等特性&#xff0c;适用于离线和在线的消息消费&#xff0c;如常规的…

内网隧道代理技术(十五)之 Earthworm的使用(二级代理)

Earthworm的使用(二级代理) 本文紧接着上一篇文章继续讲解Earthworm工具的使用 (二级代理)正向连接 二级正向代理发生在如下的情况: 1、Web服务器在公网,黑客可以直接访问 2、B机器在内网,黑客不能直接访问 3、Web服务器可以访问内网机器B 4、内网机器B可以访问公司…

ARM将常数加载到寄存器方法之LDR伪指令

一、是什么&#xff1f; LDR Rd,const伪指令可在单个指令中构造任何32位数字常数,使用伪指令可以生成超过MOV和MVN指令 允许范围的常数. 实现原理: (1)如果可以用MOV或MVN指令构造该常数,则汇编程序会生成适当的指令 (2)如果不能用MOV或MVN指令构造该常数,则汇编程序会执行下列…

【UE5】快速认识入门

目录 &#x1f31f;1. 快速安装&#x1f31f;2. 简单快捷键操作&#x1f31f;3. 切换默认的打开场景&#x1f31f;4. 虚幻引擎术语 &#x1f31f;1. 快速安装 进入Unreal Engine 5官网进行下载即可&#xff1a;UE5 &#x1f4dd;官方帮助文档 打开后在启动器里创建5.2.1引擎…

Vue2 第七节 Vue监测数据更新原理

&#xff08;1&#xff09;Vue会监视data中所有层次的数据 &#xff08;2&#xff09;如何监测对象中的数据 通过setter实现监视&#xff0c;且要在new Vue时传入要监测的数据对象中后追加的属性&#xff0c;Vue默认不做响应式处理如果要给后添加的属性做响应式&#xff0c;使…

【雕爷学编程】MicroPython动手做(18)——掌控板之声光传感器2

知识点&#xff1a;什么是掌控板&#xff1f; 掌控板是一块普及STEAM创客教育、人工智能教育、机器人编程教育的开源智能硬件。它集成ESP-32高性能双核芯片&#xff0c;支持WiFi和蓝牙双模通信&#xff0c;可作为物联网节点&#xff0c;实现物联网应用。同时掌控板上集成了OLED…

小研究 - 基于解析树的 Java Web 灰盒模糊测试(二)

由于 Java Web 应用业务场景复杂, 且对输入数据的结构有效性要求较高, 现有的测试方法和工具在测试Java Web 时存在测试用例的有效率较低的问题. 为了解决上述问题, 本文提出了基于解析树的 Java Web 应用灰盒模糊测试方法. 首先为 Java Web 应用程序的输入数据包进行语法建模创…

【C++】模板

前言 在我们平时的代码中经常会有不同类型的变量去执行效果差不多的函数。比如&#xff1a;swap(交换)&#xff0c;sort(排序)。这些函数里其实会有大部分重复的段落&#xff0c;在这种情况下我们会使用重载函数&#xff0c;但是函数重载会有如下的问题&#xff1a; 1. 重载的函…

测试开源C#人脸识别模块ViewFaceCore(4:口罩检测、性别预测、年龄预测)

ViewFaceCore模块中的MaskDetector类支持识别人脸是否戴了口罩或有遮挡&#xff0c;主要调用PlotMask函数执行口罩检测操作&#xff0c;其函数原型如下所示&#xff1a; PlotMaskResult PlotMask<T>(T image, FaceInfo info)public class PlotMaskResult{//// 摘要:// …