【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"/…

计蒜客3月普及组

Tutorial of Popularziation A 题出的不好 12 点到 1 点不会相交&#xff0c;24 点不计算 void solve(){int x, y, res;cin >> x >> y;res y - x;if(x < 11 && y > 12) res --;if(y 24) res --;cout << res << \n; }B 题解有问题…

使用mybatis拦截器日志打印sql执行时间(yml配置开关)

1.yml配置开关 monitoring:sql : true 2.拦截器实现 /*** author qujingye* Classname SqlStatementInterceptor* Description sql时间监控* Date 2024/4/3 14:56*/ Intercepts({Signature(type Executor.class, method "update", args {MappedStatement.class…

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、将文件存储到转存的目录中 文件下载…

大宋咨询(深圳商业地产调查)房地产消费者问卷调查

面对复杂多变的地产市场&#xff0c;了解消费者的需求和偏好是至关重要的。通过进行消费者问卷调查&#xff0c;房地产开发商和营销人员可以收集到宝贵的数据&#xff0c;从而做出更明智的决策。下面将详细介绍大宋咨询&#xff08;深圳问卷调查公司&#xff09;如何进行房地产…

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

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

Java面试题:Java集合框架:请简述Java集合框架的主要组成部分,并解释它们之间的关系。

Java集合框架&#xff08;Java Collections Framework&#xff09;是一组用来表示和操作集合的类的集合&#xff0c;它提供了用于存储不同类型对象的标准化接口和类。Java集合框架的主要组成部分包括以下几个部分&#xff1a; 集合接口&#xff08;Collection Interface&#…

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请求都是独立的…

【WPF应用30】WPF中的ListBox控件详解

WPF&#xff08;Windows Presentation Foundation&#xff09;是.NET框架的一个组成部分&#xff0c;用于构建桌面应用程序的用户界面。ListBox是WPF中一个非常常用的控件&#xff0c;用于显示一系列的项&#xff0c;用户可以选择单个或多个项。 1.ListBox的基本概念 ListBox…

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

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

Neo4j基础知识

图数据库简介 图数据库是基于数学里图论的思想和算法而实现的高效处理复杂关系网络的新型数据库系统。它善于高效处理大量的、复杂的、互连的、多变的数据。其计算效率远远高于传统的关系型数据库。 在图形数据库当中&#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网络。该仪表/传感器组合提供的测量包括峰值功率…

算法训练day56leetcode300.最长递增子序列674最长连续递增序列

part13leetcode300.最长递增子序列674最长连续递增序列 718最长重复子数组 300. 最长递增子序列 本题最关键的是要想到dp[i]由哪些状态可以推出来&#xff0c;并取最大值&#xff0c;那么很自然就能想到递推公式&#xff1a;dp[i] max(dp[i], dp[j] 1); #include <iost…