【Leetcode】top 100 回溯

基础知识补充

回溯中的组合问题:

优化:剪枝:在for循环时需要根据当前状态调整循环次数(组合问题)

基础操作补充

!!!牢记模板!!!

result = []
def backtrack(选择列表, 路径):if 满足结束条件:result.add(路径)returnfor 选择 in 选择列表:# 做选择路径.add(选择)将该选择从选择列表移除backtrack(选择列表, 路径) # 核心 递归调用之前【做选择】,调用之后【撤销选择】# 撤销选择路径.remove(选择)将该选择再加入选择列表
题目 
46 全排列

给定一个不含重复数字的数组 nums ,返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。

class Solution:def permute(self, nums):res = []# 选择列表就是nums包含的元素# 使用used标记已经选择的数字 间接表示选择列表的变化def backtrack(path, used):# 结束条件if len(path) == len(nums):res.append(path[:]) # !!!此处有坑需要注意returnfor i in range(len(nums)):if used[i]:    # nums[i]已经选过 跳过continue# 做选择path.append(nums[i])used[i] = True # 更新选择列表# 递归backtrack(path, used)# 撤销选择path.pop()used[i] = False # 回退选择列表的变化# 初始时路径为空,所有元素都没有选择过所以used中都是Falseused = [False]*len(nums)path = []backtrack(path, used)return res
78 子集

给你一个整数数组 nums ,数组中的元素 互不相同 。返回该数组所有可能的子集(幂集)。解集 不能 包含重复的子集。你可以按 任意顺序 返回解集。

class Solution(object):def subsets(self, nums):""":type nums: List[int]:rtype: List[List[int]]"""res = []def backtrack(path, start):res.append(path[:])if len(path) == len(nums):return for i in range(start, len(nums)):path.append(nums[i])     # 递归backtrack(path, i+1)   # 从下一元素开始 避免重复path.pop()path, start = [], 0backtrack(path, start)return res
17 电话号码的字母组合

给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。

digits.length = 0: return []

digits.length = 1: return ['x','y','z']

digits.length = k: k个组合中各挑一个字母组合

class Solution(object):def letterCombinations(self, digits):""":type digits: str:rtype: List[str]"""tmp = {'2':'abc', '3':'def', '4':'ghi', '5':'jkl', '6':'mno', '7': 'pqrs', '8':'tuv', '9':'wxyz'}strr = []for i in digits:strr.append(tmp[i])length = len(strr)if length == 0: return []elif length == 1: return [i for i in strr[0]]res = []def backtrack(path, start):if len(path) == len(strr):res.append(''.join(path))return for i in range(start, len(strr)):for j in range(len(strr[i])):path.append(strr[i][j])     backtrack(path, i+1)path.pop()path, start = [], 0backtrack(path, start)return res
39 组合总和

给你一个 无重复元素 的整数数组 candidates 和一个目标整数 target ,找出 candidates 中可以使数字和为目标数 target 的 所有 不同组合 ,并以列表形式返回。你可以按 任意顺序 返回这些组合。candidates 中的 同一个 数字可以 无限制重复被选取 。如果至少一个数字的被选数量不同,则两种组合是不同的。 对于给定的输入,保证和为 target 的不同组合数少于 150 个。

candidates = [1,2,3]  tartget = 5     

会出现[1,1,3]-[1,3,1]的重复情况,因此需要将可选范围限制在当前值(可重复选取)或之后;

class Solution(object):def combinationSum(self, candidates, target):""":type candidates: List[int]:type target: int:rtype: List[List[int]]"""res = []def backtrack(path, start):summ = sum(path)if summ >= target:                          # 终止条件不再限制长度if summ== target:res.append(path[:])return for i in range(start, len(candidates)):path.append(candidates[i])     backtrack(path, i)                      # 当前元素和之后的可重复选取path.pop()path, start = [], 0backtrack(path, start)return res
22 括号生成

数字 n 代表生成括号的对数,请你设计一个函数,用于能够生成所有可能的并且 有效的 括号组合。

‘(’代表+1,‘)’代表-1,终止条件:sum(path)出现负数时结束,len(path)==2*n时结束;

class Solution(object):def generateParenthesis(self, n):""":type n: int:rtype: List[str]"""res = []inputt = ['(', ')']def backtrack(path, summ):if summ < 0: return if len(path) == 2*n: if summ == 0: res.append(''.join(path))return for i in range(len(inputt)):path.append(inputt[i])   summ += (1 if inputt[i]=='(' else -1)  backtrack(path, summ)                    strr = path.pop()summ -= (1 if strr=='(' else -1)path, summ = [], 0backtrack(path, summ)return res

DFS:记录左括号和右括号的数量;

class Solution:def generateParenthesis(self, n):res = []def dfs(paths, left, right):if left > n or right > left: returnif len(paths) == n * 2:  # 因为括号都是成对出现的res.append(paths)returndfs(paths + '(', left + 1, right)  # 生成一个就加一个dfs(paths + ')', left, right + 1)dfs('', 0, 0)return res
79 单词搜索

给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中,返回 true ;否则,返回 false 。单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母不允许被重复使用。

先遍历一次,找到所有字符串单词首字母在网格中的位置;

从首字母的网格位置入手,先加入当前字母,然后在不越界的情况下加入上/下/左/右的字母,判断加入的第n个字母是否和word中第n个字母相同;

class Solution(object):def exist(self, board, word):""":type board: List[List[str]]:type word: str:rtype: bool"""res = []m, n, k =  len(board), len(board[0]), len(word)if m*n < k: return False              # 元素不够def backtrack(path, cnt, pos):if path[-1] != word[cnt-1]: returnelse:if cnt == k:res.append(path[:])return for i,j in [[-1,0],[1,0],[0,-1],[0,1]]:x, y = pos[0]+i, pos[1]+jif -1<x<m and -1<y<n and board[x][y]:path.append(board[x][y])cnt, board[x][y] = cnt+1, ''     backtrack(path, cnt, [x, y])   # 从下一元素开始 避免重复board[x][y] = path.pop()cnt -= 1first_alpha = []for i in  range(m):for j in range(n):if board[i][j] == word[0]:first_alpha.append([i,j])if len(first_alpha)==0: return False  # 没有首字母for i,j in first_alpha:path, cnt = [board[i][j]], 1board[i][j] = ''backtrack(path, cnt, [i, j])board[i][j] = word[0]return bool(res)

可以优化的地方:1.不需要记录path;2.找到一个word就可以结束;(需要backtrack回传)3.改变传入的cnt即可;

class Solution(object):def exist(self, board, word):""":type board: List[List[str]]:type word: str:rtype: bool"""m, n, k =  len(board), len(board[0]), len(word)if m*n < k: return False              # 元素不够def backtrack(cnt, pos):if board[pos[0]][pos[1]] != word[cnt]: return Falseif cnt == k-1: return Trueboard[pos[0]][pos[1]] = ''for i,j in [[-1,0],[1,0],[0,-1],[0,1]]:x, y = pos[0]+i, pos[1]+jif -1<x<m and -1<y<n and board[x][y]:    if backtrack(cnt+1, [x, y]): return Trueboard[pos[0]][pos[1]] = word[cnt]first_alpha = []for i in  range(m):for j in range(n):if board[i][j] == word[0]:first_alpha.append([i,j])if len(first_alpha)==0: return False  # 没有首字母for i,j in first_alpha:if backtrack(0, [i, j]): return Truereturn False

 131 分割回文串

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

子串的选择范围需要从下一字符开始,因此需要起始位置;

加入path的子串必须是回文子串,需要先判断再加入,因此把枚举范围从单个字符变成字符串长度,然后验证当前长度的子串是否是回文;

终止条件:长度到达终点;

class Solution(object):def partition(self, s):""":type s: str:rtype: List[List[str]]"""res = []length = len(s)def backtrack(path, start):if start == length:res.append(path[:]) return for i in range(start, length):    # 枚举的是回文子串的长度tmp = s[start:i+1] if tmp==tmp[::-1]:path.append(tmp)backtrack(path, i+1)path.pop()path, start = [], 0backtrack(path, start)return res
 51 N皇后

按照国际象棋的规则,皇后可以攻击与之处在同一行或同一列或同一斜线上的棋子。

n 皇后问题 研究的是如何将 n 个皇后放置在 n×n 的棋盘上,并且使皇后彼此之间不能相互攻击。给你一个整数 n ,返回所有不同的 n 皇后问题 的解决方案。每一种解法包含一个不同的 n 皇后问题 的棋子放置方案,该方案中 'Q' 和 '.' 分别代表了皇后和空位。

搬运【山鬼】的解析:

1. 从上往下放棋子,用变量的表示就是,row从0到n-1
2.backtrack()函数的参数为row,表示当前开始放第row行的皇后,小于row的行(row行往上的)都已经放置皇后了
3. 需要构建一个isValid()函数,参数为board,row,col,返回是否可以在row,col上合法放置皇后;这个函数需要沿着row,col判断这个位置的上方、右上方、左上方是否有皇后。
 

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

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

相关文章

UE4_破碎插件的蓝图节点_Apply Radius Damage

一、知识点 Apply Radius Damage:破碎组件所带的蓝图节点。 二、使用方法&#xff1a; 1、设置——插件&#xff0c;搜索destruction&#xff0c;找到 Apex Destruction&#xff0c;勾选已启用。重启虚幻编辑器。 2、这样右键操作就有创建可破坏的网格体菜单&#xff0c;将do…

八、从0开始卷出一个新项目之瑞萨RZN2L 3.1.7 debug调试和下载

目录 3.1.7 debug调试和下载 3.1.7.1 官方介绍 3.1.7.2 e2studio debug变量实时监控 3.1.7.3 Iar debug变量实时监控 3.1.7.4 debug经验总结 八、从0开始卷出一个新项目之瑞萨RZN2L 3.1.7 debug调试和下载 3.1.7 debug调试和下载 3.1.7.1 官方介绍 官网&#xff1a; d…

【国信华源2024年首场春季校园招聘面试会举办】

阳春三月&#xff0c;春意盎然&#xff0c;北京国信华源科技有限公司2024年校园招聘活动如期展开。4月2日&#xff0c;成功举办了“国信华源2024年首场春季校园招聘面试会”。 国信华源公司人力资源部热情接待了前来参加面试的同学们&#xff0c;并亲自陪同他们深入探访了企业。…

实操:driver.js 实现产品导览、亮点、上下文帮助

官网 https://driverjs.com/ 依赖 <script src"https://cdn.jsdelivr.net/npm/driver.js1.0.1/dist/driver.js.iife.js"></script> <link rel"stylesheet" href"https://cdn.jsdelivr.net/npm/driver.js1.0.1/dist/driver.css"/…

Apache DolphinScheduler 【安装部署】

前言 今天来学习一下 DolphinScheduler &#xff0c;这是一个任务调度工具&#xff0c;现在用的比较火爆。 1、安装部署 1.0、准备工作 1.0.1、集群规划 dolphinscheduler 比较吃内存&#xff0c;所以尽量给 master 节点多分配一点内存&#xff0c;桌面和虚拟机里能关的应用…

Spring Boot--文件上传和下载

文件上传和下载 前言文件上传1、以MultipartFile 接口流文件&#xff0c;流的名称需要和前台传过来的名称对应上2、获取到文件名称截取后缀3、为了放置文件名重复使用uuid来随机生成id后缀4、判断转存路径中是否有这个文件夹如果没有就创建5、将文件存储到转存的目录中 文件下载…

Android移动应用与开发上机实验报告

实验目的&#xff1a; 本项目需要开发一个Android App&#xff0c;运行后显示“欢迎XXX学习Android开发(第1行)、祝学有所成、马到成功&#xff01;(第2行)”。 根据该实验需求与实现思路(P26-27)&#xff0c;在获得素材的基础上&#xff0c;对手机主界面写代码进行实现&…

SpringBoot参数校验@Valid 和 @Validated注解使用详解

JSR-303 是 JAVA EE 6 中的一项子规范&#xff0c;叫做 Bean Validation&#xff0c;官方参考实现是Hibernate Validator。 注意&#xff1a;JSR-303实现与 Hibernate ORM 没有任何关系。 JSR 303 用于对 Java Bean 中的字段的值进行验证。 Spring MVC 3.x 之中也大力支持 JS…

web框架的本质初识

1.什么是HTML HTML是一个超文本语言&#xff0c;是一种创建网页结构的标记语言。就是你女朋友化妆之后的样子 2.什么是HTTP协议 是一种用于在Web上传输数据的协议。它是客户端和服务器之间进行相互通信的基础的协议 3.HTTP的特点 无连接&#xff1a;每个http请求都是独立的…

择校!这些计算机专业的考研学校性价比巨高(必看)

建议可以关注一下东北大学&#xff0c;可以抄底 今年东北大学刚更改408&#xff0c;加上地区不太优势&#xff0c;很可能爆冷&#xff0c;有时候会觉得学校的选择可能比个人的努力更加重要。要做出明智的选择&#xff0c;需要考虑近几年的复试分数线&#xff0c;以及当年的热度…

C 练习实例97 - 读磁盘 写磁盘

题目&#xff1a;从键盘输入一些字符&#xff0c;逐个把它们送到磁盘上去&#xff0c;直到输入一个‘#’为止 在桌面新建一个hello.txt文件&#xff0c;内容示例&#xff1a; 代码&#xff1a; #include <stdio.h> #include <stdlib.h>int main() {FILE *fp; //文…

详解k8s集群内外的访问方式

文章目录 1、集群内访问2、集群外访问2.1、Ingress转发外网请求2.2、LoadBanlancer接入外网请求2.3、NodePort接入外网请求 3、总结和对比3.1、Ingress、NodePort和LoadBalancer总结3.2、Ingress和网关的区别 1、集群内访问 在k8s中创建的微服务&#xff0c;大部分都是在集群内…

N1912A安捷伦N1912A功率计

181/2461/8938产品概述&#xff1a; 安捷伦N1912A双通道P系列宽带功率传感器为R&D和制造工程师提供精确和可重复的功率测量&#xff0c;应用市场包括航空航天和国防&#xff08;雷达&#xff09;、无线通信和无线802.11a/b/g网络。该仪表/传感器组合提供的测量包括峰值功率…

c++|vector使用及模拟实现

目录 一、vector的介绍 二、vector的使用(常用接口) 2.1string类的成员函数 2.1.1构造函数 2.1.2析构函数 2.1.3“”运算符重载函数 2.2 迭代器(iterator) 及 对象的遍历访问 2.2.1iterator 2.2.2 operator[] && at() 2.2.4 back() && front() 2.2…

Springboot自动获取接口实现

ServiceLoader加载接口实现步骤 1.编写接口 public interface CommunicationAdapterFactory {void setKernel(LocalKernel kernel);boolean providesAdapterFor(Vehicle vehicle);BasicCommunicationAdapter getAdapterFor(Vehicle vehicle); }2.编写实现 // 实现类 1 publi…

计算机网络:数据链路层 - 点对点协议PPP

计算机网络&#xff1a;数据链路层 - 点对点协议PPP PPP协议的帧格式透明传输字节填充法零比特填充法 差错检测循环冗余校验 对于点对点链路&#xff0c;PPP协议是目前使用最广泛的数据链路层协议。比如说&#xff0c;当用户想要接入互联网&#xff0c;就需要通过因特网服务提供…

【随笔】Git 高级篇 -- 分离 HEAD(十一)

&#x1f48c; 所属专栏&#xff1a;【Git】 &#x1f600; 作  者&#xff1a;我是夜阑的狗&#x1f436; &#x1f680; 个人简介&#xff1a;一个正在努力学技术的CV工程师&#xff0c;专注基础和实战分享 &#xff0c;欢迎咨询&#xff01; &#x1f496; 欢迎大…

C 回调函数的两种使用方法

对回调&#xff08;callback&#xff09;函数的一点粗陋理解&#xff0c;在我小时候&#xff0c;隔壁村有家月饼小作坊&#xff08;只在中秋那段时间手工制作一些月饼出售&#xff0c;后来好像不做了&#xff09;&#xff0c;做出的月饼是那种很传统很经典的款式&#xff0c;里…

C++项目——集群聊天服务器项目(十三)客户端登录、注册、退出业务

截止到上节&#xff0c;我们已将服务器端主要代码介绍完毕&#xff0c;由于不可能一直手动输入信息&#xff0c;所以我们还需编写客户端代码&#xff0c;进行双向通信。 客户端不要求高并发&#xff0c;因此我们这里不使用muduo网络库的TcpClient类编写&#xff0c;仅采用C自带…

Lumos学习王佩丰Excel第一讲:认识Excel

最近发现自己在操作excel的一些特殊功能时会有些不顺手&#xff0c;所以索性找了一个比较全的教程&#xff08;王佩丰excel24讲&#xff09;拿来学习&#xff0c;刚好形成文档笔记&#xff0c;分享给有需要但没有时间看视频的朋友们。整体笔记以王老师授课的知识点去记录&#…