刷题总结 回溯算法

news2025/1/27 8:44:46

为了方便复习并且在把算法忘掉的时候能尽量快速的捡起来

刷完回溯算法这里需要做个总结

回溯算法的适用范围

回溯算法是深度优先搜索(DFS)的一种特定应用,在DFS的基础上引入了约束检查回退机制

相比于普通的DFS,回溯法的优势主要体现在解决需要约束条件判断、剪枝回退的复杂问题上。

实际应用时有易于识别的题目类型特征组合、分割、子集、排列、棋盘

这些问题的共同点是解空间均为层次结构,因此回溯算法就是对回溯树进行DFS

模式识别

根据自己刷题的感受,并统计做题过程中思路上遇到过的槛,

归纳后可以得到以下模式识别树,

专门用于看到题目后进行模式识别,

快速找到解题方法:

该模式识别树综合搜索集类型、搜索规则和结果收集条件等因素

接下来对当前的模式识别树进行简单解释:

回溯算法总模板:

def backtracking(self, 参数):
    if 终止条件
        存放结果
        return

    for 选择:本层集合中元素(树中节点孩子的数量就是集合的大小):
        处理节点
        self.backtracking(路径,选择列表) // 递归
        回溯,撤销处理结果

步骤:1.函数(无返回值,参数多) + 2.if(终止){收集+return} + 3.for(搜索集){处理;递归;回溯}

回溯树:

一、题目类型

回溯算法常用于解决问题:组合、分割、子集、排列、棋盘等

在我眼里以上题目类型可以总结为三个大类:组合类问题、排列类问题和约束满足类问题,

因此,这几个大类是根据解空间(回溯树)的层次结构进行划分的,

而且对模板的应用有较大的影响

1.组合类问题:层内层间顺序访问

包含类型:组合、分割、子集

模板变体:

def backtracking(self, 输入参数, start_idx, path, ans):
    if 终止条件
        存放结果
        return

    for i in range(start_idx, len(总搜索集)):
        if 约束条件:
            continue # 有时需要break
        处理节点
        self.backtracking(输入参数, start_idx + 1, path, ans) // 递归
        回溯,撤销处理结果

def backtracking(self, 输入参数, start_idx, path, ans):
    if 终止条件
        存放结果
        return

    for i in range(start_idx, len(总搜索集)):
        if 约束条件:
            处理节点
            self.backtracking(输入参数, start_idx + 1, path, ans) // 递归
            回溯,撤销处理结果

模式识别特征:

  • 不考虑顺序

  • 一般搜索集长度大于结果长度

  • 回溯树层内层间顺序访问

题目列表:

组合

77. 组合

216. 组合总和 III

39. 组合总和

40. 组合总和 II

分割

131. 分割回文串

93. 复原 IP 地址

子集

78. 子集

90. 子集 II

491. 非递减子序列

这其中有一个特例就是重复选取题:39. 组合总和

递归命令需要改为:

# self.backtracking(输入参数, start_idx + 1, path, ans) // 递归
self.backtracking(输入参数, start_idx, path, ans) // 递归

2.排列类问题:层内遍历访问层间顺序访问

包含类型:排列

模板变体:

该类型题目有visited数组写法、集合写法和交换元素三种写法,

visited数组普适性较强,以visited数组写法为例:

def backtracking(self, 输入参数, visited, path, ans):
    if 终止条件
        存放结果
        return

    for i in range(len(总搜索集)):
        if visited[i] or 其他约束条件:
            continue
        处理节点
        visited[i] = True
        self.backtracking(输入参数, visited, path, ans) // 递归
        visited[i] = False
        其他回溯,撤销处理结果

模式识别特征:

  • 考虑顺序

  • 搜索集长度等于结果长度

  • 回溯树层内遍历,层间顺序访问

题目列表:

46. 全排列

47. 全排列 II

3.约束满足问题:约束条件较强,无特定访问顺序

包含类型:安排行程、棋盘

相比于总模板无特定要求,但常需要主函数预处理搜索集以剪枝回溯树,实现高效搜索

模式识别特征:

  • 一般类似于排列问题

  • 有较强的约束条件

  • 一般需要对搜索集预处理,否则会超时或代码过于复杂

题目列表:

332. 重新安排行程

棋盘

51. N 皇后

37. 解数独

二、几个需要注意的题目类型的约束条件

这个只有一个维度,目前共遇到三个种类:

1.组合:结果收集条件和剪枝优化

1.1 长度

用长度作为结果收集条件会产生两个结果:

(1)路径长度符合条件时收集结果并返回

(2)剪枝优化:搜索集末尾剪枝

可以通过控制单个节点内下一步访问的搜索集的末尾位置

实现方式未

回溯树的末端指针从n调整到n - (k - len(path)) + 1

题目:

77. 组合

理解起来就是把末尾的(k - len(path))个路径去掉,

以长度作为结果收集条件的题目总搜索集长度 > 路径长度 + 节点内搜索集宽度

节点内搜索集宽度太大了,路径长度就达不到要求,

所以需要把回溯树的宽度从n调整到n - (k - len(path)) + 1,

末尾的+1是本轮访问的节点

1.2 求和值大小

类似于长度,也是有两个影响:

(1)路径求和等于目标数时收集结果并返回,大于目标数直接返回

(2)剪枝优化:排序break剪枝,

将搜索集排序,求和大于目标数舍弃节点所有后续路径

题目:

216. 组合总和 III

39. 组合总和

40. 组合总和 II

实现方式为:

如果下一层的sum(就是本层的 sum + candidates[i])已经大于target,就可以结束本轮for循环的遍历

注意主函数种要对总搜索集排序

2.分割:分割片段的约束条件

分割类题目对分割片段有特殊的要求,因此常采用以下模板范式:

def is_valid(self, 参数):
    判断条件

def backtracking(self, 输入参数, start_idx, path, ans):
    if 终止条件
        存放结果
        return

    for i in range(start_idx, len(总搜索集)):
        if self.is_valid(参数):
            处理节点
            self.backtracking(输入参数, start_idx + 1, path, ans) // 递归
            回溯,撤销处理结果

题目:

131. 分割回文串

93. 复原 IP 地址

3.其他特殊约束条件:棋盘、安排行程等

该类题,常常需要复杂的判断条件或在主函数中对搜索集预处理(常用哈希表)来辅助判断约束条件

其中复杂的判断条件可能导致超时,

在主函数中对搜索集预处理(常用哈希表)来辅助判断比较常用

题目:(预处理方式)

安排行程:map哈希表:

332. 重新安排行程

棋盘:数组哈希表:

51. N 皇后

37. 解数独

三、去重

1.组合问题中的去重

如果搜索集存在重复元素,则需要进行同层重复剪枝操作来去重,

否则由于回溯树中同一层存在重复元素,使得多条路径通向同一个结果

导致重复的组合结果,

去重方法可以分为索引去重数值去重,但都需要排序

其中排序有两个作用:使相同数字紧贴防止异层重复访问

防止异层重复访问是去重能够在层内进行的前提,

索引去重便是利用相同数字紧贴的特性去重,有简单写法数组写法两种写法

数值去重则是在层内统计同一个数值的使用情况来去重,不需要排序的第一个功能,

更详细介绍在刷题记录 回溯算法-13:90. 子集 II-CSDN博客

题目:

40. 组合总和 II

90. 子集 II

491. 非递减子序列

其中491. 非递减子序列情况特殊,只能用数值去重,详见刷题记录 回溯算法-14:491. 非递减子序列-CSDN博客

2.排列问题中的去重

和组合问题略有不同,排列问题除了层内去重还可以倒序去重,

层内去重逻辑和组合去重类似,索引去重数值去重都可以用,同样需要排序

但索引去重的简单写法无法使用,只能用数组写法

倒序去重很少用,以上详见:

刷题记录 回溯算法-16:47. 全排列 II-CSDN博客

题目:47. 全排列 II

四、遍历与搜索(递归函数返回)

大部分题目需要找到所有结果,但有些题目找到一个结果即可返回,其代码结构会出现一些变化

1.收集所有结果

大部分题目都需要用一个数组收集所有符合条件的结果,

这也是总模板适配的情况

def backtracking(self, 参数):
    if 终止条件
        存放结果
        return

    for 选择:本层集合中元素(树中节点孩子的数量就是集合的大小):
        处理节点
        self.backtracking(路径,选择列表) // 递归
        回溯,撤销处理结果

2.找到一个结果

有些题目只需要找到一个结果即可,且往往尝试找到所有结果会导致超时,

这种情况需要用返回值在找到结果后快速返回主函数:

def backtracking(self, 参数):
    if 终止条件
        存放结果
        return True

    for 选择:本层集合中元素(树中节点孩子的数量就是集合的大小):
        处理节点
        if self.backtracking(路径,选择列表):  # 递归
            return True
        回溯,撤销处理结果
    return False

修改分为三处:

1.终止条件返回True

2.递归调用返回True

3.函数末尾返回False 

题目:

332. 重新安排行程

37. 解数独

建议的复习方法

回溯算法类型的题目有较为统一的模板和容易识别的题目类型,

但各个类型都会有格子需要注意的点,

因此建议先简单过一遍模式识别树,熟悉模板和思路

然后按题目类型顺序逐个类型过一遍算法题,强化各个类型的特定模式

当遇到不熟悉的模式识别环节,就做一下特定模式识别环节的强化练习

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

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

相关文章

Antd React Form使用Radio嵌套多个Select和Input的处理

使用Antd React Form使用Radio会遇到嵌套多个Select和Input的处理&#xff0c;需要多层嵌套和处理默认事件和冒泡&#xff0c;具体实现过程直接上代码。 实现效果布局如下图 代码 <Formname"basic"form{form}labelWrap{...formItemLayoutSpan(5, 19)}onFinish{on…

Linux(Centos、Ubuntu) 系统安装jenkins服务

该文章手把手演示在Linux系统下如何安装jenkins服务、并自定义jenkins数据文件位置、以及jenkins如何设置国内镜像源加速&#xff0c;解决插件下载失败问题 安装方式&#xff1a;war包安装 阿里云提供的war下载源地址&#xff1a;https://mirrors.aliyun.com/jenkins/war/?s…

【基于无线电的数据通信链】Link 11 仿真测试

〇、废话 Link 11 仿真测试 涉及多个方面&#xff0c;包括信号仿真、协议模拟、数据链路层的仿真以及网络性能评估等。Link 11 是一种基于 HF&#xff08;高频&#xff09; 或 UHF&#xff08;超高频&#xff09; 波段的无线通信协议&#xff0c;主要用于军事通信系统中。为了…

VScode 开发 Springboot 程序

1. 通过maven创建springboot程序 输入 mvn archetype:generate 选择模板&#xff0c;一般默认选择为第 7 种方式&#xff1b; 选择之后&#xff0c;一般要你填写如下内容&#xff1a; groupId: 组织名称&#xff1b;artifactId: 项目名称&#xff1b;version: 版本&#xff0…

深入MapReduce——引入

引入 前面我们已经深入了HDFS的设计与实现&#xff0c;对于分布式系统也有了不错的理解。 但HDFS仅仅解决了海量数据存储和读写的问题。要想让数据产生价值&#xff0c;一定是需要从数据中挖掘出价值才行&#xff0c;这就需要我们拥有海量数据的计算处理能力。 下面我们还是…

springfox-swagger-ui 3.0.0 配置

在3.0中&#xff0c;访问地址URL变了。 http://地址:端口/项目名/swagger-ui/ SpringBoot maven项目引入 <dependency><groupId>io.springfox</groupId><artifactId>springfox-swagger2</artifactId><version>3.0.0</version> </…

如何解压7z文件?8种方法(Win/Mac/手机/网页端)

7z 文件是一种高效的压缩文件格式&#xff0c;由 7 - Zip 软件开发者所采用。它运用独特的压缩算法&#xff0c;能显著缩小文件体积&#xff0c;便于存储与传输各类数据&#xff0c;像软件安装包、大型资料集等。但要使用其中内容&#xff0c;就必须解压&#xff0c;因为处于压…

Spring Boot 邂逅Netty:构建高性能网络应用的奇妙之旅

一、引言 在当今数字化时代&#xff0c;构建高效、可靠的网络应用是开发者面临的重要挑战。Spring Boot 作为一款强大的 Java 开发框架&#xff0c;以其快速开发、简洁配置和丰富的生态支持&#xff0c;深受广大开发者喜爱。而 Netty 作为高性能、异步的网络通信框架&#xff…

[STM32 - 野火] - - - 固件库学习笔记 - - -十一.电源管理系统

一、电源管理系统简介 电源管理系统是STM32硬件设计和系统运行的基础&#xff0c;它不仅为芯片本身提供稳定的电源&#xff0c;还通过多种电源管理功能优化功耗、延长电池寿命&#xff0c;并确保系统的可靠性和稳定性。 二、电源监控器 作用&#xff1a;保证STM32芯片工作在…

从 Web2 到 Web3:技术演进中的关键变革

随着互联网的快速发展&#xff0c;Web 技术经历了从 Web1 到 Web2&#xff0c;再到当前热议的 Web3 的演变。每一次技术迭代不仅仅是技术本身的升级&#xff0c;更代表着对社会、经济和文化的深刻影响。本文将带你走过 Web2 到 Web3 的技术演进&#xff0c;探讨其中的关键变革&…

Android实战经验篇-玩转Selinux(详解版)

列文章转如下链接&#xff1a; Android Display Graphics系列文章-汇总 Android实战经验篇-系列文章汇总 本文主要包括部分&#xff1a; 一、Selinux概述 1.1 SELinux是什么&#xff1f; 1.2 自主访问控制&#xff08;DAC&#xff09; 1.3 强制访问控制&#xff08;MAC&…

CLOUDFLARE代理请求重定向你太多次

现象 使用CLOUDFLARE代理前请求正常&#xff0c;使用CLOUDFLARE代理请求后出现 原因分析 以下是我的猜测&#xff0c;在默认情况下 CLOUDFLARE代理&#xff0c;可能是直接请求我们服务器的IP&#xff0c;比如&#xff1a;http://1.1.1.1 而不是通过域名的方式&#xff08;如…

U-Net - U型网络:用于图像分割的卷积神经网络

U-Net是一种专为图像分割任务设计的卷积神经网络&#xff08;CNN&#xff09;&#xff0c;最初由Olaf Ronneberger等人于2015年提出。它被广泛应用于医学影像分析、遥感图像分割、自动驾驶和其他许多需要对图像进行像素级分类的任务中。U-Net具有强大的特征提取和恢复能力&…

第十五届蓝桥杯大赛软件赛省赛C/C++ 大学 B 组

第十五届的题目在规定时间内做出了前5道&#xff0c;还有2道找时间再磨一磨。现在把做的一些思路总结如下&#xff1a; 题1&#xff1a;握手问题 问题描述 小蓝组织了一场算法交流会议&#xff0c;总共有 50人参加了本次会议。在会议上&#xff0c;大家进行了握手交流。按照惯例…

Vue3 + TS 实现批量拖拽 文件夹和文件 组件封装

一、html 代码&#xff1a; 代码中的表格引入了 vxe-table 插件 <Tag /> 是自己封装的说明组件 表格列表这块我使用了插槽来增加扩展性&#xff0c;可根据自己需求&#xff0c;在组件外部做调整 <template><div class"dragUpload"><el-dial…

DX12 快速教程(4) —— 画钻石原矿

快速导航 新建项目 "004-DrawTexture"纹理贴图纹理采样纹理过滤邻近点采样双线性过滤Mipmap 多级渐远纹理三线性过滤各向异性过滤 纹理环绕LOD 细节层次 开始画钻石原矿吧加载纹理到内存中&#xff1a;LoadTexture什么是 WIC如何用 WIC 读取一帧图片获取图片格式并转…

LBS 开发微课堂|AI向导接口服务:重塑用户的出行体验

为了让广大开发者 更深入地了解 百度地图开放平台的 技术能力 轻松掌握满满的 技术干货 更加简单地接入 位置服务 我们特别推出了 “位置服务&#xff08;LBS&#xff09;开发微课堂” 系列技术案例 第六期的主题是 《AI向导接口服务的能力与接入方案》 随着地图应…

mysql 学习3 SQL语句--整体概述。SQL通用语法;DDL创建数据库,查看当前数据库是那个,删除数据库,使用数据库;查看当前数据库有哪些表

SQL通用语法 SQL语句分类 DDL data definition language : 用来创建数据库&#xff0c;创建表&#xff0c;创建表中的字段&#xff0c;创建索引。因此成为 数据定义语言 DML data manipulation language 有了数据库和表以及字段后&#xff0c;那么我们就需要给这个表中 添加数…

【Bug 记录】el-sub-menu 第一次进入默认不高亮

项目场景&#xff1a; 项目场景&#xff1a;el-sub-menu 第一次进入默认不高亮 问题描述 例如&#xff1a;sub-menu 的 index 后端默认传过来是 number&#xff0c;我们需要手动转为 string&#xff0c;否则会有警告&#xff0c;而且第一次进入 sub-menu 默认不高亮。 解决方…

深入探讨ncnn::Mat类——ncnn中的核心数据结构

最近在学习 ncnn 推理框架&#xff0c;下面整理了 ncnn::Mat 的使用方法。 ncnn作为一个高性能的神经网络推理框架&#xff0c;其核心数据结构ncnn::Mat在数据存储与处理上扮演了至关重要的角色。本文将从基础到高级&#xff0c;详细介绍ncnn::Mat类的各个方面&#xff0c;帮助…