递归与回溯 || 排列问题

news2024/11/24 7:12:24

目录

前言:

全排列

题解:

全排列 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;
        }

        //递归出口,后面再怎么加也不能凑出target
        if(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.coloradmin.cn/o/1836652.html

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

相关文章

MySQL数据库回顾(1)

数据库相关概念 关系型数据库 概念: 建立在关系模型基础上&#xff0c;由多张相互连接的二维表组成的数据库。 特点&#xff1a; 1.使用表存储数据&#xff0c;格式统一&#xff0c;便于维护 2.使用SQL语言操作&#xff0c;标准统一&#xff0c;使用方便 SOL SQL通用语法 …

MySQL常见面试题自测

文章目录 MySQL基础架构一、说说 MySQL 的架构&#xff1f;二、一条 SQL语句在MySQL中的执行过程 MySQL存储引擎一、MySQL 提供了哪些存储引擎&#xff1f;二、MySQL 存储引擎架构了解吗&#xff1f;三、MyISAM 和 InnoDB 的区别&#xff1f; MySQL 事务一、何谓事务&#xff1…

JCR一区 | Matlab实现GAF-PCNN、GASF-CNN、GADF-CNN的多特征输入数据分类预测/故障诊断

JJCR一区 | Matlab实现GAF-PCNN、GASF-CNN、GADF-CNN的多特征输入数据分类预测/故障诊断 目录 JJCR一区 | Matlab实现GAF-PCNN、GASF-CNN、GADF-CNN的多特征输入数据分类预测/故障诊断分类效果格拉姆矩阵图GAF-PCNNGASF-CNNGADF-CNN 基本介绍程序设计参考资料 分类效果 格拉姆…

数据结构与算法-字符出现的次数

问题描述 以下是这个找出字符串中字符串出现频率最多的字符。大家可以自行研究一下&#xff0c;题目不难&#xff0c;我今天尝试使用C语言来完成解答&#xff0c;但是在解答过程居然出现了一个意想不到的问题。可能是高级语言用多了&#xff0c;C语言某些函数的限制和风险忘记管…

Android开发系列(三)Jetpack Compose 之TextField

TextField 是一个用于接收用户输入的UI组件。它是Jetpack Compose中的一部分&#xff0c;可以方便地实现用户文本输入的功能。 TextField 允许用户输入一个或多个文本行&#xff0c;可以用于接收用户的文本输入、搜索等操作。它提供了一些常用的功能&#xff0c;如输入验证、键…

深入了解SD-WAN:企业广域网的未来

在讨论SD-WAN之前&#xff0c;我们先来了解一下WAN的基本概念。WAN&#xff08;广域网&#xff09;是一个连接多个地理位置分散的局域网的通信网络。在企业中&#xff0c;WAN通常连接总部、分支机构、托管设施和云服务等多个网络节点。广域网允许用户共享各种应用和服务&#x…

LeetCode 算法:合并两个有序链表 c++

原题链接&#x1f517;&#xff1a;合并两个有序链表 难度&#xff1a;简单⭐️ 题目 将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。 示例 1&#xff1a; 输入&#xff1a;l1 [1,2,4], l2 [1,3,4] 输出&#xff1a;…

20240616日志:大模型压缩方法DMS

Location: Beijing 1 大模型剪枝 Fig. 1.1大模型压缩-剪枝 剪枝的理论来源基于彩票假设&#xff08;Lottery Ticket Hypothesis&#xff09;&#xff0c;指在神经网络中存在一种稀疏连接模式&#xff0c;即仅利用网络的一小部分连接&#xff08;彩票&#xff09;就足以实现与整…

论坛产品选型,需要关注哪些点?

论坛社区是一个经久不衰的行业&#xff0c;比如我们常见的宠物社区&#xff0c;校园社区&#xff0c;游戏社区、企业内部社区&#xff0c;品牌社区&#xff0c;本地同城、私域社区项目、付费社群、问答社区等等&#xff0c;可以说是覆盖了各行各业&#xff0c;那么如果我们要搭…

设备档案包括哪些内容

设备档案通常包括以下内容和要求&#xff1a; 1. 设备基本信息&#xff1a;包括设备名称、型号、规格、生产厂商、出厂日期、购买日期等。 2. 设备安装信息&#xff1a;包括设备的安装位置、安装日期、安装人员等。 3. 设备维护信息&#xff1a;包括设备的维护保养记录&#xf…

构建基于 LlamaIndex 的RAG AI Agent

I built a custom AI agent that thinks and then acts. I didnt invent it though, these agents are known as ReAct Agents and Ill show you how to build one yourself using LlamaIndex in this tutorial. 我构建了一个自定义的AI智能体&#xff0c;它能够思考然后行动。…

C/C++ string模拟实现

1.模拟准备 1.1因为是模拟string&#xff0c;防止与库发生冲突&#xff0c;所以需要命名空间namespace隔离一下&#xff0c;我们来看一下基本内容 namespace yx {class string{private://char _buff[16]; lunix下小于16字节就存buff里char* _str;size_t _size;size_t _capac…

文字炫酷祝福 含魔法代码

效果下图&#xff1a;&#xff08;可自定义显示内容&#xff09; 代码如下&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initi…

深度学习(十)——神经网络:非线性激活

一、Padding Layers简介 nn.ZeroPad2d&#xff1a;在输入的tensor数据类型周围用0进行填充 nn.ConstantPad2d&#xff1a;在输入的tensor数据类型周围用常数进行填充 这个函数的主要作用是对输入的图像进行填充&#xff0c;但里面所有功能都能用nn.Conv2d实现。 二、Non-li…

CAD二次开发(10)-单行文字的添加+图形修改

1. 单行文字的添加 第一步&#xff1a; 首先在CAD中新增中文样式 输入ST命令&#xff1a; 第二步&#xff1a;代码开发 /// <summary>/// 添加文本信息/// </summary>[CommandMethod("AddText")]public void AddText(){var doc Application.DocumentM…

分析医药零售数据该用哪个BI数据可视化工具?

数据是企业决策的重要依据&#xff0c;可以用于现代企业大数据可视化分析的BI工具有很多&#xff0c;各有各擅长的领域。那么哪个BI数据可视化工具分析医药零售数据又好又快&#xff1f; 做医药零售数据分析首推奥威BI数据可视化工具&#xff01; 奥威BI数据可视化工具做医药…

c++实现二叉搜索树(下)

好久不见啊&#xff0c;baby们&#xff0c;小吉我又回归了&#xff0c;发完这一篇小吉将会有两周时间不会更新blog了&#xff08;sorry&#xff09;&#xff0c;在小吉没有发blog的日子里大家也要好好学习数据结构与算法哦&#xff0c;还有就是别忘了小吉我❤️  这篇博客是二…

新火种AI|Sora发布半年之后,AI视频生成领域风云再起

作者&#xff1a;一号 编辑&#xff1a;美美 AI视频最近有些疯狂&#xff0c;Sora可能要着急了。 自OpenAI的Sora发布以来&#xff0c;AI视频生成技术便成为了科技界的热门话题。尽管Sora以其卓越的性能赢得了广泛关注&#xff0c;但其迟迟未能面向公众开放&#xff0c;让人…

计算机网络——传输层重要协议(TCP、UDP)

一、常见名词解释 IP地址&#xff1a;IP地址主要用于标识网络主机、其他网络设备&#xff08;如路由器&#xff09;的网络地址&#xff0c;即IP地址用于定位主机的网络地址&#xff1b; IP地址是一个32位的二进制数&#xff0c;通常被分割为4个 8位⼆进制数&#xff08;也就是…

C++之std::type_identity

目录 1.简介 2.C20的std::type_identity 3.使用 type_identity 3.1.阻止参数推导 3.1.1.模板参数推导过程中的隐式类型转换 3.1.2.强制显式实例化 3.2.阻止推断指引 3.3.类型保持 3.4.满足一些稀奇古怪的语法 4.示例 5.总结 1.简介 std::type_identity 是 C17 引入的…