递归与回溯 || 排列问题

目录

前言:

全排列

题解:

全排列 II

 题解:

子集

题解: 

 组合

题解:

组合总和

题解:

电话号码的字母组合

题解: 

字母大小写全排列

题解:

优美的排列

题解: 


前言:

递归与回溯问题需要弄清楚以下几点:

1、递归前需要做什么?

2、什么时候递归,什么时候回溯?

3、回溯时需要做什么,需要返回值吗,如何接收返回值,需要恢复现场吗,还是什么都不需要处理?

全排列

46. 全排列 - 力扣(LeetCode)

题解:

面对这种排列问题,首先需要画出决策树,根据决策树来实现代码

我们以示例一为例,决策树如下:

nums = [ 1, 2, 3 ], 假设我们选择排列的第一个数为 1,我们继续在 1 2 3 里面选择排列的第二个数,由于 1 已经被选过了,我们只能选择 2 或 3 作为排列的第二个数:

1、假设选择 2 为排列的第二个数继续在 1 2 3 里面选择排列的第三个数,由于 1 2 已经被选过了,我们只能选择 3 作为排列的第三个数,最终排列的结果为 1 2 3 

2、假设选择 3 为排列的第二个数继续在 1 2 3 里面选择排列的第三个数,由于 1 3 已经被选过了,我们只能选择 2 作为排列的第三个数,最终排列的结果为 1 3 2 。

选择 2 作为排列的第一个数也是同理。结合文字和图中的决策树可以看出,我们每次都会在 1 2 3 里面做选择,但每次选择时,会排除已经选过的数字(因为不能重复选)。

nums = [ 1, 2, 3, 4 ] 也是同样的道理(只截取一部分作为参考):

接下来我们需要回答开头提出的几个问题:

1、递归前需要做什么?

递归前需要找出排列的下一个数字。我们用 for 循环来模拟决策树做选择的过程,为了避免选择了重复的数字,我们用 bool 数组来标记,true 表示该数字已经选择过了,false 表示该数字还没有被选择过。如果找到了排列的下一个数字,把该数字添加到排列中,并把该数字标记为 true。

为了便于把数字尾插到排列中,我们把排列设计为全局变量。

2、什么时候递归?

找到排列的下一个数字后,就可以递归,寻找排列的下一个数。

3、什么时候回溯?

排列后的数组长度 == nums 的长度时,就可以回溯。

4、回溯时需要做什么?

以 1 2 3 4 的排列为例,我们找到一个排列 1 2 3 4 之后(红色路径),需要从递归的最后一层回到 2 这一层,再选择 4 这个数字,继续递归(橙色路径),找出排列 1 2 4 3。这就要求我们:

1、从递归的最后一层 4 回溯到倒数第二层 3 时,把刚刚尾插到排列的数字 4 删掉,且把 4 置为 false(我们称这一操作为恢复现场),这样就可以把排列恢复到 1 2 3,然后继续回溯;

2、从递归的倒数第二层 3 回溯到 2 这一层时,当前排列为 1 2 3,我们把排列的最后一个数字删掉,并把 3 置为 false,这样就可以把排列恢复为 1 2,这样就可以接着递归橙色路径,找到排列 1 2 4 3。

总结:

回溯时需要   1、把当前排列的最后一个数字删掉;  2、把该数字置为 false 。

class Solution {vector<int> path;vector<vector<int>> ret;bool check[7];
public:vector<vector<int>> permute(vector<int>& nums) {dfs(nums);return ret;}void dfs(vector<int>& nums){if(path.size()==nums.size())//递归出口{ret.push_back(path);return;}for(int i=0;i<nums.size();i++)//模仿决策过程{if(check[i]==false)//该数字未访问过{path.push_back(nums[i]);check[i]=true;//把该数字设为访问过dfs(nums);//继续递归path.pop_back();//回溯时恢复现场check[i]=false;//把该数字恢复为未访问过}}}
};

全排列 II

47. 全排列 II - 力扣(LeetCode)

 题解:

这道题比较麻烦的是处理重复元素的排列,我们假设 nums = [ 1 ,1 ,1 ,2 ]:

从部分决策树可以看出, 即使我们标记了哪个元素已经访问过了,依旧会出现重复的排列!所以我们需要观察得出排列的规律(先对 nums 进行排序,这样重复的数就会排在一起,便于讨论):

如果 nums[ i ] 还没有被访问过

1、nums[ i ] 排在数组的第一个位置,那么 nums[ i ] 可以添加到排列;

2、 nums[ i ] 虽然不是数组的第一个元素,但是 nums[ i ] 和 nums[ i -1 ] 不相等,说明 nums[ i ] 可能在数组中只出现了一次,或者出现了很多次,但是 nums[ i ] 是这堆重复元素中第一个出现的,可以添加到排列中;

3、 nums[ i ] 不是数组的第一个元素,和 nums[ i -1 ] 相等了,但是 nums[ i -1 ] 已经被访问过了,那么可以添加到排列中。

因为递归是根据数组下标按从小到大的顺序添加到排列中的,若 nums [ i ] == nums[ i -1 ],对于同一层递归中,在访问  nums[ i ] 之前, nums[ i -1 ] 一定已经递归结束,且已经得出排列的结果了,而   nums[ i ]  和  nums[ i -1 ] 递归得到的排列是相同的,所以  nums[ i ] 没有必要进行递归了,所以剪枝!

class Solution {vector<vector<int>> ret;vector<int> path;bool check[9];
public:vector<vector<int>> permuteUnique(vector<int>& nums) {sort(nums.begin(),nums.end());dfs(nums,0);return ret;}void dfs(vector<int>& nums,int pos){if(pos==nums.size())//回溯 {ret.push_back(path);return;}for(int i=0;i<nums.size();i++){//当前数字为false,或者这个数字是第一个数字,或者这个数字和前一个不相同,//或者这个数字和前一个相同,但是前一个数字为true,则可以继续递归if(check[i]==false && (i==0 || nums[i]!=nums[i-1] || check[i-1]==true)){path.push_back(nums[i]);check[i]=true;dfs(nums,pos+1);path.pop_back();//恢复现场check[i]=false;}}}
};

子集

78. 子集 - 力扣(LeetCode)

题解: 

决策树如下:

在挑选子集的时候,由于子集的无序性,子集 [ 1, 2 ] 和 [ 2, 1 ] 是相同的集合,为了避免结果中出现元素相同但顺序不同的集合,我们需要规定,在找子集时,不要回头去访问比子集的第一个元素小的数字。

比如决策树中,我们从 2 开始找子集,那我们就从 2 往后寻找元素,不要回头去访问比 2 小的数了,最终找到的子集就是 [ 2 ] , [ 2 , 3 ],从 3 开始找子集,就从 3 往后寻找元素,不要回头去访问比 3 小的数,最终找到的子集就是 [ 3 ] 。  

为了实现这一规定,递归时需要记录上一层访问的数字,记为 pos,在 for 循环里面寻找元素时,从 pos 开始往后找。这其实是一个剪枝的操作!剪去了不必要的访问!

class Solution {vector<int> path;//子集vector<vector<int>> ret;
public:vector<vector<int>> subsets(vector<int>& nums) {ret.push_back(path);//空集dfs(nums,0);return ret;}void dfs(vector<int>& nums,int pos){if(pos==nums.size())    return;for(int i=pos;i<nums.size();i++){path.push_back(nums[i]);ret.push_back(path);dfs(nums,i+1);path.pop_back();}}
};

 组合

77. 组合 - 力扣(LeetCode)

题解:

和子集类似,为了避免出现重复的组合,需要记录上一层访问的数字 start,用 for 循环寻找组合的元素时,只需要从 start 往后开始寻找,不要回头访问数字!

class Solution {vector<int> path;vector<vector<int>> ret;bool check[21];
public:vector<vector<int>> combine(int n, int k) {dfs(n,k,1);return ret;}void dfs(int n,int k,int start){if(path.size()==k){ret.push_back(path);return;}for(int i=start;i<n+1;i++){path.push_back(i);dfs(n,k,i+1);path.pop_back();}}
};

组合总和

39. 组合总和 - 力扣(LeetCode)

题解:

决策树如下:

这道题规定一个数可以被无限重复次使用,所以我们不需要标记元素是否被访问过。

但是会出现重复的组合,比如 [ 2 , 2 , 3 ] 和 [ 3 , 2 , 2 ] 的组合总和都是 target,但是组合的元素相同,只是顺序不同,这样的组合就是重复的。为了避免出现重复的组合,我们要记录上一层访问的元素 pos,用 for 循环选择组合元素时,从 pos 往后开始选择,避免挑选组合的元素时走回头路。

这道题还需要注意递归的出口:

1、当 组合总和 ==  target 时,这个组合就是我们想要的组合,把该组合添加到结果数组,return;

2、如果 组合总和 > target ,已经没有继续寻找组合元素的必要了,return;

3、如果 组合总和 < target ,但是 组合总和+ candidates[ 0 ]  > target (candidates 已排序),即 目前的组合总和 加上 candidates 最小的数 就已经超过 target,那么 目前的组合总和 无论加上 candidates 的哪个数,最终结果一定会大于 target,此时已经没有继续寻找的必要了,return。

class Solution {vector<vector<int>> ret;vector<int> path;int pathsum=0;
public:vector<vector<int>> combinationSum(vector<int>& candidates, int target) {sort(candidates.begin(),candidates.end());if(candidates[0]>target)    return ret;dfs(candidates,target,0);return ret;}void dfs(vector<int>& candidates,int target,int pos){if(pathsum==target)//进结果{sort(path.begin(),path.end());ret.push_back(path);               return;}//递归出口,后面再怎么加也不能凑出targetif(pathsum>target || pathsum+candidates[0]>target) return;for(int i=pos;i<candidates.size();i++){path.push_back(candidates[i]); pathsum+=candidates[i];dfs(candidates,target,i);//i决定了不会走回头路path.pop_back(); pathsum-=candidates[i];//恢复现场}}
};

电话号码的字母组合

17. 电话号码的字母组合 - 力扣(LeetCode)

题解: 

class Solution {vector<string> tel{"","","abc","def","ghi","jkl","mno","pqrs","tuv","wxyz"};string path;vector<string> ret;
public:vector<string> letterCombinations(string digits) {if(digits.size()==0) return ret;dfs(digits,0);return ret;}void dfs(const string& digits,int pos){if(pos==digits.size()){ret.push_back(path);return;}for(auto ch:tel[digits[pos]-'0'])//访问数字对应的字母{path.push_back(ch);//pos+1,访问下一个数字dfs(digits,pos+1);path.pop_back();//恢复现场}}
};

字母大小写全排列

784. 字母大小写全排列 - 力扣(LeetCode)

题解:

这道题的决策树稍微有点不一样,有点类似二叉树,左子树是不变,右子树是变。

由于只需要改变大小写字母,在走 变 的这条分支时,如果当前访问的字符串为字母时,才需要大小写转换。

class Solution {vector<string> ret;string path;
public:vector<string> letterCasePermutation(string s) {dfs(s,0);return ret;}void dfs(const string& s,int pos){if(pos==s.size()){ret.push_back(path);return;}//不改变path.push_back(s[pos]);dfs(s,pos+1);path.pop_back();//改变if(s[pos]<'0' || s[pos]>'9'){char ch=change(s[pos]);path.push_back(ch);dfs(s,pos+1);path.pop_back();}}char change(char ch){if(ch>='a' && ch<='z') ch-=32;else ch+=32;return ch;}
};

优美的排列

526. 优美的排列 - 力扣(LeetCode)

题解: 

class Solution {int ret=0;bool check[16];
public:int countArrangement(int n) {dfs(n,1);return ret;}void dfs(int n,int i)//i为下标,pos为perm[i]{if(i==n+1){++ret;return;}for(int pos=1;pos<=n;pos++){if(!check[pos] && (pos%i==0 || i%pos==0)){check[pos]=true;dfs(n,i+1);check[pos]=false;}}}
};

未完待续,欢迎读者指出文章的错误!

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

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

相关文章

AI虚拟数字人上线需要办理哪些资质?

近年来&#xff0c;随着AI 技术快速发展&#xff0c;虚拟数字人行业也进入了新的发展阶段。AI 技术可覆盖虚拟数字人的建模、视频生成、驱动等全流程&#xff0c;一方面使虚拟数字人的制作成本降低、制作周期缩短&#xff0c;另一方面&#xff0c;多模态 AI 技术使得虚拟数字人…

[面试题]缓存

[面试题]Java【基础】[面试题]Java【虚拟机】[面试题]Java【并发】[面试题]Java【集合】[面试题]MySQL[面试题]Maven[面试题]Spring Boot[面试题]Spring Cloud[面试题]Spring MVC[面试题]Spring[面试题]MyBatis[面试题]Nginx[面试题]缓存[面试题]Redis 什么是缓存&#xff1f;…

KVM虚拟化基础

虚拟化基础与分类 x86 CPU特权级别设为四个Ring&#xff1a;Kernel Mode运行在Ring 0、User Mode运行在Ring 3&#xff1b; 非硬件辅助虚拟化&#xff08;一型&#xff09; Hypervisor运行在Ring 0&#xff1b; Guest OS不做修改&#xff0c;以为自己运行在Ring 0上&#xff0…

OpenAPI Typescript Codegen 的基本使用

下载 axios npm install axios OpenAPI Typescript Codegen 官网&#xff1a;https://github.com/ferdikoomen/openapi-typescript-codegen 安装 OpenAPI Typescript Codegen npm install openapi-typescript-codegen --save-dev–input&#xff1a;指定接口文档的路径、url …

小程序餐饮点餐系统,扫码下单点菜,消费端+配送端+收银端+理端

目录 前言&#xff1a; 一、小程序功能有哪些 前端&#xff1a; 管理端&#xff1a; 二、实体店做小程序的好处 方便快捷的点餐和支付体验&#xff1a; 扩大店铺的曝光度和影响力&#xff1a; 优化顾客体验和服务质量&#xff1a; 降低成本和提高效率&#xff1a; 数据…

【学习】程序员资源网址

1 书栈网 简介&#xff1a;书栈网是程序员互联网IT开源编程书籍、资源免费阅读的网站&#xff0c;在书栈网你可以找到很多书籍、笔记资源。在这里&#xff0c;你可以根据热门收藏和阅读查看大家都在看什么&#xff0c;也可以根据技术栈分类找到对应模块的编程资源&#xff0c;…

Microsoft Visual C++ Redistributable 【安装包】【高速下载】

方法1、可以从官方下载&#xff0c;如下图 Visual C Redistributable for Visual Studio 2015 但是此链接只有一个版本 方法2 已经下载好并且已经整理好了2008--2022的所有版本点击下方链接即可高速下载 如果是win7-win8-win10-win11直接可以下载2015--2022版本&#xff0c…

大模型基础知识:探索人工智能的巨轮

人工智能大模型&#xff0c;这个在近年来频繁出现在科技新闻和学术论坛的热门词汇&#xff0c;已经成为了推动人工智能技术发展的关键力量。这些大模型&#xff0c;如OpenAI的GPT-3、谷歌的BERT、百度的ERNIE等&#xff0c;以其强大的性能和广泛的应用范围&#xff0c;引起了广…

【AI大模型】在健康睡眠监测中的深度融合与实践案例

文章目录 1. 应用方案2. 技术实现2.1 数据采集与预处理2.2 构建与训练模型2.3 个性化建议生成 3. 优化策略4. 应用示例&#xff1a;多模态数据融合与实时监测4.1 数据采集4.2 实时监测与反馈 5. 深入分析模型选择和优化5.1 LSTM模型的优势和优化策略5.2 CNN模型的优势和优化策略…

若依RuoYi-Vue分离版—配置多数据源

若依RuoYi-Vue分离版—配置多数据源 一、修改application-druid.yml二、修改pom文件&#xff0c;引入依赖第一种&#xff1a;下载jar包到本地&#xff0c;然后引入&#xff08;我这边用的是这种&#xff09;本地引入的&#xff0c;打包时需要加上配置 第二种&#xff1a;从远程…

随想录Day63 | 单调栈 42. 接雨水 84.柱状图中最大的矩形

随想录Day63 | 单调栈 42. 接雨水 84.柱状图中最大的矩形 42. 接雨水 题目链接 42 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图&#xff0c;计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 第一次提交 class Solution { public:int trap(vector<int>…

云原生微服务开发日趋成熟:有效拥抱左移以改善交付

在软件工程和应用程序开发方面&#xff0c;云原生已经成为许多团队的常用术语。当人们调查云原生的世界时&#xff0c;他们经常会得出这样的观点&#xff1a;云原生的整个过程都是针对大型企业应用程序的。几年前&#xff0c;情况可能确实如此&#xff0c;但随着 Kubernetes 等…

《Fundamentals of Power Electronics》——理想变压器基本公式推导

接下去推导理想变压器的基本公式。理想变压器满足以下三个条件&#xff1a; 1、无铜损。假设原副边线圈均无纯电阻&#xff0c;则不会因在铜导线中产生焦耳热引起能量损耗&#xff0c;另外也不考虑回路中的分布电容。 2、无铁损。忽略通过铁芯的磁通量变化引起的涡流损耗&…

线性二次型调节器(LQR)举例

线性二次型调节器(LQR) 线性二次型调节器(LQR)是一种用于最优控制的问题,其中目标是通过最小化某个代价函数来找到最优控制策略。LQR特别适用于线性系统。为了在人形机器人上应用LQR进行建模,主要步骤包括建立系统模型、定义代价函数以及求解最优控制律。以下是详细步骤…

Chromium 开发指南2024 Mac篇-Xcode安装(二)

1.引言 在开始编译和开发 Chromium 之前&#xff0c;确保开发环境的正确配置是至关重要的。对于 Mac 用户来说&#xff0c;Xcode 是不可或缺的工具&#xff0c;因为它不仅提供了必需的编译器和工具链&#xff0c;还包含了与 macOS 系统深度整合的开发资源。在本系列指南的第一…

Apache Druid-时序数据库

Apache Druid&#xff1a;是是一个集时间序列数据库、数据仓库和全文检索系统特点于一体的分析性数据平台&#xff0c;旨在对大型数据集进行快速的查询分析&#xff08;"OLAP"查询)。Druid最常被当做数据库来用以支持实时摄取、高性能查询和高稳定运行的应用场景&…

微信商城源码带分销功能 支持微信小程序+公众号端 含详细图文搭建教程

分享一款微商城源码系统&#xff0c;不仅支持微信公众号端和小程序端二合一管理&#xff0c;带分销机制&#xff0c;集合了市面上电商系统的功能&#xff0c;并增加了多项全新特色功能&#xff0c;如多商户商家入驻、区域代理、收银台、互动直播等&#xff0c;为商家提供了全方…

多行文本的文字展示全部和收起功能

组件代码&#xff1a; <template><!-- 外层容器&#xff0c;使用相对定位 --><div class"relative"><!-- 文本容器&#xff0c;根据 expanded 状态决定是否应用 line-clamp-4 类 --><div :class"{ line-clamp-4: !expanded }"…

表达式的格式化

表达式&#xff1a;一条具有明确结果的代码语句 在字符串前面加上前缀f&#xff0c;表示这是一个格式化字符串&#xff0c;可以在字符串中直接引用变量&#xff0c;并使用{}来表示这些变量的位置。例如&#xff1a; name "Alice" age 30 print(f"My name is …

scratch3编程02-使用克隆来编写小游戏

目录 1&#xff0c;游戏效果 2&#xff0c;游戏代码块 1&#xff09;玩家 2&#xff09;障碍物 ​ 3&#xff09;箭头 ​ 4&#xff09;关卡图片 3&#xff0c;scratch文件 1&#xff0c;游戏效果 使用克隆 在这个游戏中&#xff1a; 程序开始&#xff1a;只要点击“…