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

news2024/10/6 8:35:24

题目 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+1
                ans.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.coloradmin.cn/o/812551.html

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

相关文章

【多模态】21、BARON | 通过引入大量 regions 来提升模型开放词汇目标检测能力

文章目录 一、背景二、方法2.1 主要过程2.2 Forming Bag of Regions2.3 Representing Bag of Regions2.4 Aligning bag of regions 三、效果 论文&#xff1a;Aligning Bag of Regions for Open-Vocabulary Object Detection 代码&#xff1a;https://github.com/wusize/ovdet…

彻底理解Java中的各种锁

1、乐观锁 乐观锁是一种乐观思想&#xff0c;假定当前环境是读多写少&#xff0c;遇到并发写的概率比较低&#xff0c;读数据时认为别的线程不会正在进行修改&#xff08;所以没有上锁&#xff09;。写数据时&#xff0c;判断当前 与期望值是否相同&#xff0c;如果相同则进行…

linux(centos) docker 安装 nginx

​1、拉取nginx最新版本镜像 docker pull nginx:latest 查看镜像 docker images 或者 docker images -a 2.启动nginx容器 docker run -d -p 80:80 --name nginx nginx 使用docker run命令&#xff0c;启动nginx容器。 --name&#xff0c;设置容器名。为方便记忆&#xff…

【TypeScript】TS入门及基础学习(一)

【TypeScript】TS入门及基础学习&#xff08;一&#xff09; 【TypeScript】TS入门及基础学习&#xff08;一&#xff09;一、前言二、基本概念1.强类型语言和弱类型语言2.动态语言和静态语言 三、TypeScript与JavaScript的区别四、环境搭建及演练准备4.1 安装到本地4.2 在线运…

【移动机器人运动规划】01 —— 常见地图基础 |图搜索基础

文章目录 前言相关代码整理:相关文章&#xff1a; 可视化网址&#xff1a;常用地图基础Occupancy grid mapOcto-mapVoxel hashingPoint cloud mapTSDF mapESDF mapFree-space RoadmapVoronoi Diagram Map 图搜索基础配置空间图搜索基本概念DijkstraAStarAstar的一些变种&#x…

CAN转ETHERCAT网关将CAN 总线和 ETHERCAT 网络连接方法

由于好多现场会出现将CAN总线的设备接到EtherCAT网络中&#xff0c;由于协议的不相同&#xff0c;不能直接进行连接&#xff0c;现需一种能同时兼容CAN 总线和ETHERCAT网络的一种设备&#xff0c;由此捷米JM-ECT-CAN 是自主研发的一款 ETHERCAT 从站功能的通讯网关。该产品主要…

深入浅出:大语言模型中必不可少的技术——Embedding简介

今天&#xff0c;推特上一位科技博主SullyOmarr分享了一个关于embedding的内容十分火爆。主要介绍为什么embedding对于在目前的AI大模型中很重要。这是一个十分不错的关于embedding知识的介绍。本文将根据SullyOmarr的内容也对embedding做一个简单的介绍&#xff0c;并解释为什…

Linux启动流程详解

Linux 开机启动流程 Linux 启动顺序是指 Linux 系统从开机到进入用户登录界面的过程&#xff0c;它可以分为以下几个步骤&#xff1a; 加载 BIOS&#xff1a;BIOS 是基本输入输出系统&#xff0c;它负责检测硬件设备&#xff0c;设置启动顺序&#xff0c;读取第一个启动设备的…

Java小型操作系统模拟(采用策略模式结合反射进行搭建,支持一些简单的命令)

Java小型操作系统模拟 项目说明第一阶段&#xff1a;反射结合策略模式搭建基本的命令结构第二阶段&#xff1a;注解结合反射与策略模式&#xff0c;将结构进一步规范第三阶段&#xff1a;开启新的窗口&#xff0c;将控制台输入切换到新窗口中&#xff0c;同时创建右键菜单&…

如何理解Diffusion

Diffusion算法可以有多个角度进行理解&#xff0c;不同的理解方式只是对目标函数进行了不同的解释。其主体思想是不变的&#xff0c;可以归纳为&#xff1a; 训练时通过图片逐步添加噪声&#xff0c;变为一个纯噪声。然后学习每一步的噪声。推理时给定一个随机噪声图片&#x…

Leetcode | Binary search | 22. 74. 162. 33. 34. 153.

22. Generate Parentheses 要意识到只要还有左括号&#xff0c;就可以放到path里。只要右括号数量小于左括号&#xff0c;也可以放进去。就是valid的组合。recurse两次 74. Search a 2D Matrix 看成sorted list就好。直接用m*n表示最后一位的index&#xff0c;并且每次只需要 …

Android启动速度优化

本节主要内容&#xff1a;了解APP启动流程、启动状态、查看启动时间、CPU Profile定位启动耗时代码、StrictMode严苛模式检测不合理写法、解决启动黑白屏问题。 一、APP启动流程 ①用户点击桌面App图标&#xff0c;Launcher进程采用Binder IPC向system_server进程发起startAc…

Spring Boot集成Swagger3.0,Knife4j导出文档

文章目录 Spring Boot集成Swagger3.0,Knife4j导出文档效果展示如何使用简要说明添加依赖添加配置类测试接口token配置位置 官网 说明情况 demo Spring Boot集成Swagger3.0,Knife4j导出文档 效果展示 如何使用 简要说明 Knife4j的前身是swagger-bootstrap-ui,前身swagger-boo…

day48-Random Image Feed(随机图片显示)

50 天学习 50 个项目 - HTMLCSS and JavaScript day48-Random Image Feed&#xff08;随机图片显示&#xff09; 效果 index.html <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport&…

msvcp120.dll丢失的解决方法?有什么解决方法比较推荐?

第一&#xff1a;介绍几种可能导致msvcp120.dll文件丢失或损坏的原因 损坏的程序安装&#xff1a;在安装某个程序时&#xff0c;可能会出现意外中断或其他错误&#xff0c;导致msvcp120.dll文件未能正确地被安装或被破坏。这可能是由于软件安装过程中的错误、病毒感染或硬件问题…

OSPF协议RIP协议+OSPF实验(eNSP)

本篇博客主要讲解单区域的ospf&#xff0c;多区域的仅作了解。 目录 一、OSPF路由协议概述 1.内部网关协议和外部网关协议 二、OSPF的应用环境 1.从以下几方面考虑OSPF的使用 2.OSPF的特点 三、OSPF重要基本概念 3.1&#xff0c;辨析邻居和邻接关系以及七种邻居状态 3…

BPMNJS插件使用及汉化(Activiti绘制流程图插件)

BPMNJS插件运行最重要的就是需要安装nodejs插件,这不一定要安装和测试好。 主要是使用npm命令 1、配置BPMNJS插件绘制activiti7工作流 1.1、安装和配置nodejs 插件 1.1.1、下载nodejs 下载地址:https://nodejs.org/en 1.1.2、安装nodejs,傻瓜式安装 安装之后在安装…

【etcd】docker 启动单点 etcd

etcd: v3.5.9 etcd-browser: rustyx/etcdv3-browser:latest 本文档主要描述用 docker 部署单点的 etcd&#xff0c; 用 etcd-browser 来查看注册到 etcd 的 key 默认配置启动 docker run -d --name ai-etcd --networkhost --restart always \-v $PWD/etcd.conf.yml:/opt/bitn…

一文搞懂mysql(安装、基础命令、存储文件)

1、安装 除此之外&#xff0c;windows在安装前需要额外补加两个东西 dxwebsetup.exe、 vcredist_x64.exe 这俩随便一搜就能找到 在安装前者时要注意取消勾选bing工具栏 mysql下载链接 2、初始化 管理员身份打开cmd >> "path_to_mysql/bin/msqld.exe" --initi…

Windows11+Opencv+Clion编译源码

Windows11OpencvClion编译源码 参考&#xff1a;https://www.robotsfan.com/posts/69395e08.html 注意事项 编译过程中使用的软件&#xff0c;开源码等所有工具的安装路径一定不要有中文和空格。cmake过程会下载一些文件&#xff0c;如果是局域网的话可能下载不下来&#xf…