穷举vs暴搜vs深搜vs回溯vs剪枝(一)

news2025/1/15 8:33:12

文章目录

  • 全排列
  • 子集
  • 找出所有子集的异或总和再求和
  • 全排列 II
  • 电话号码的字母组合

全排列

题目:全排列

在这里插入图片描述

思路

通过深度优先搜索的方式,不断枚举每个数在当前位置的可能性,然后回溯到上一个状态,直到枚举完所有可能性得到正确的结果

在这里插入图片描述

  • res:一个二维向量vector<vector<int>>,用于存储所有生成的排列。
  • path:一个一维向量vector<int>,用于存储当前正在构建的排列。
  • check:一个布尔数组bool check[7],用于标记数组nums中的每个元素是否已经被用于当前排列中。这里假设nums数组的大小不会超过6(因为数组索引从0开始,最大索引为6时,数组大小为7)。
  • 递归的终止条件是path的大小等于nums的大小。
  • 在递归过程中,使用check数组来确保不会重复使用同一个元素。
  • 使用path.push_back(nums[i])path.pop_back()来实现回溯,即在尝试下一个元素之前,需要将当前元素从path中移除,以便尝试其他可能的元素组合。
  • 通过check[i] = truecheck[i] = false来标记元素是否已被使用。

C++代码

class Solution 
{
    vector<vector<int>> res;
    vector<int> path;
    bool check[7];

public:
    void dfs(vector<int>& nums)
    {
        if(nums.size() == path.size())
        {
            res.push_back(path);
            return ;
        }

        for(int i = 0; i < nums.size(); i ++ )
        {
            if(!check[i])
            {
                path.push_back(nums[i]);
                check[i] = true;
                dfs(nums);
                // 回溯 -> 恢复现场
                path.pop_back();
                check[i] = false;
            }
        }
    }

    vector<vector<int>> permute(vector<int>& nums) 
    {
        dfs(nums);
        return res;
    }
};

子集

题目:子集

在这里插入图片描述

思路1

在这里插入图片描述

我们先将所有结果个数为1的选出来,再再其基础上选出结果个数为2的,依次类推

  • pos 开始遍历 nums 数组。对于每个位置 i ,执行以下操作:
    nums[i] 添加到 path 中。
  • 递归调用 dfs 函数,以 i + 1 作为新的起始位置,继续搜索。
  • 在递归调用返回后,进行回溯操作:将刚刚添加到 path 中的 nums[i] 移除,以便尝试其他可能的元素组合(或停止进一步搜索)。

C++代码

class Solution 
{
    vector<vector<int>> ret;
    vector<int> path;
public:
    vector<vector<int>> subsets(vector<int>& nums) 
    {
        dfs(nums, 0);
        return ret;
    }

    void dfs(vector<int>& nums, int pos)
    {
        ret.push_back(path);

        for(int i = pos; i < nums.size(); i++)
        {
            path.push_back(nums[i]);
            dfs(nums, i + 1);
            path.pop_back();
        }
    }
};

思路2

在这里插入图片描述

依次枚举,1选不选,选的话在1基础上2选不选,选的话在2基础上选不选,枚举出所有结果;当当前位置等于数组大小时,将结果加入答案中

  • 首先检查pos是否等于nums的大小。如果是,说明已经遍历到数组的末尾,此时path代表了一个完整的子集(可能是空集,也可能是包含数组所有元素的集合,这取决于dfs的调用过程)。然后,将这个子集添加到ret中,并返回
  • nums[pos]添加到path中,然后递归调用dfs函数,以pos + 1作为新的起始位置继续搜索。在递归调用返回后,执行回溯操作:从path中移除nums[pos],以便尝试其他可能的元素组合或停止进一步搜索。
class Solution 
{
    vector<vector<int>> ret;
    vector<int> path;
public:
    vector<vector<int>> subsets(vector<int>& nums) 
    {
        dfs(nums, 0);
        return ret;
    }

    void dfs(vector<int>& nums, int pos) 
    {
        if (pos == nums.size()) 
        {
            ret.push_back(path);
            return;
        }

        path.push_back(nums[pos]);
        dfs(nums, pos + 1);
        path.pop_back();

        dfs(nums, pos + 1);
    }
};

找出所有子集的异或总和再求和

题目:找出所有子集的异或总和再求和

在这里插入图片描述
思路

本题和上题思路一样,我们使用上题的第一种思路,依次枚举,1选不选,选的话在1基础上2选不选,选的话在2基础上选不选;
不同的是将每个子集的值异或,并将其相加

  • 从数组的第一个元素开始,递归地构建所有可能的子集
  • 在每个递归步骤中,可以选择包含当前元素(通过异或操作更新path)或不包含当前元素(直接递归到下一个位置)
  • int path选择1将其异或在path上,再选2异或在path上;
  • 当到达数组的末尾时,将当前的 path(即当前子集的异或和)加到 res

C++代码

class Solution 
{
    int path;
    int res;
public:
    void dfs(vector<int>& nums, int pos)
    {
        res += path;
        for(int i = pos; i < nums.size(); i ++ )
        {
            path ^= nums[i];
            dfs(nums, i + 1);
            path ^= nums[i];
        }

    }
    int subsetXORSum(vector<int>& nums) 
    {
        dfs(nums, 0);
        return res;
    }
};

全排列 II

题目:全排列 II

在这里插入图片描述
思路

同一个节点的分支中,相同的元素只能选择一次
同一个数只能使用一次


  • 只关心不合法分支

if(cheak[i] == true) || (i != 0 && nums[i] == nums[i-1] && cheak[i-1] == false))

C++代码

class Solution 
{
    vector<int> path;
    vector<vector<int>> ret; 
    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++)
        {
            if(check[i] == true|| (i != 0 && nums[i] == nums[i - 1] && check[i - 1] == false)) 
                continue;
            path.push_back(nums[i]);
            check[i] = true;
            dfs(nums, pos + 1);
            path.pop_back();
            check[i] = false;
        }

    }
};
  • 只关心合法分支

if(cheak[i] == false) && (i == 0 || nums[i] != nums[i-1] ||cheak[i-1] != false))

C++代码

class Solution 
{
    vector<int> path;
    vector<vector<int>> ret; 
    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++)
        {
            if(check[i] == false && (i == 0 || nums[i] != nums[i - 1] || check[i - 1] != false)) 
            {
                path.push_back(nums[i]);
                check[i] = true;
                dfs(nums, pos + 1);
                path.pop_back();
                check[i] = false;
            }
        }

    }
};

电话号码的字母组合

题目:电话号码的字母组合

在这里插入图片描述
思路

利用一个全局的字符串数组来帮我映射数组和字母之间的关系

  • 如果pos等于digits的长度,说明已经处理完所有数字,将当前的path(即一个完整的字母组合)添加到结果数组ret中,并返回
  • 否则,对于当前数字digits[pos],遍历其对应的所有字母(通过str[digits[pos] - '0']访问。对于每个字母,执行以下操作
    • 将当前字母添加到path的末尾。
    • 递归调用dfs函数,处理下一个数字pos + 1
    • 在递归返回后,将刚刚添加的字母从path中移除,以便尝试当前数字对应的下一个字母。

C++代码

class Solution 
{
    vector<string> ret;
    string path;
    string str[10]={"",
        "", "abc", "def",
        "ghi", "jkl", "mno",
        "pqrs", "tuv", "wxyz"
    };
public:
    vector<string> letterCombinations(string digits) 
    {
        if(digits.size() == 0) return ret;

        dfs(digits, 0);
        return ret;
    }
    void dfs(string& digits, int pos)
    {
        if(pos == digits.size())
        {
            ret.push_back(path);
            return;
        }
        for(auto ch : str[digits[pos] - '0'])
        {
            path.push_back(ch);
            dfs(digits, pos + 1);
            path.pop_back();
        }
    }
};

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

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

相关文章

FastApi SQLAlchemy SQLite

FastApi fastapi是一个用于构建API 的现代、快速&#xff08;高性能&#xff09;的web框架&#xff0c;它是建立在Starlette和Pydantic基础上的。 Pydantic是一个基于Python类型提示来定义数据验证、序列化和文档的库&#xff0c;Starlette是一种轻量级的ASGI框架/工具包&…

哪些因素会影响 FMEA 实施的效果?

在探讨哪些因素会影响FMEA&#xff08;潜在失效模式及后果分析&#xff09;实施效果的问题时&#xff0c;我们不得不深入剖析FMEA的核心理念、实施流程及其在企业质量管理中的应用实践。FMEA作为一种系统性的预防性工具&#xff0c;旨在识别产品或过程中潜在的失效模式及其影响…

C++,STL 030(24.10.14)

stack容器&#xff08;栈&#xff09;的基本概念&#xff1a; 1.stack容器是一种先进后出的数据结构&#xff0c;它只有一个出口。 2.图例&#xff1a; 注意&#xff1a; (1)进栈顺序&#xff1a;a1 -> a2 -> a3 -> a4 -> a5 (2)出栈顺序&#xff1a;a5 -> …

SVN——常见问题

基本操作 检出 提交 更新 显示日志 撤销本地修改 撤销已提交内容 恢复到指定版本 添加忽略 修改同一行 修改二进制文件

理解智能合约:区块链在Web3中的运作机制

随着区块链技术的不断发展&#xff0c;“智能合约”这一概念变得越来越重要。智能合约是区块链应用的核心之一&#xff0c;正在推动Web3的发展&#xff0c;为数字世界带来了前所未有的自动化和信任机制。本文将深入探讨智能合约的基本原理、运作机制&#xff0c;以及它在Web3生…

C++核心编程和桌面应用开发 第十天(模版 类模板)

目录 1.1函数模板语法 1.2函数模板的使用方式 1.2.1自动类型推导 1.2.2显示指定类型 1.3普通函数与模板函数 1.3.1区别 1.3.2调用规则 1.4模板的局限性 1.4.1模板的具体化 1.5类模板 1.5.1基本语法 1.5.2类模板对象做函数参数 1.5.3类模板与继承 1.5.4类模板成员…

STM32-----I2C

1.基本原理&#xff1a; 上图是I2C的总线图和通讯协议图&#xff08;就是I2C是怎么实现设备之间读写数据的&#xff09; 下面主要介绍通讯协议的每一步&#xff1a; 1.发出开始信号: 一开始都为高电平为空闲状态。当SCL为高电平时&#xff0c;主机将SDA拉低即为发出开始信号&…

hadoop集群搭建-安装虚拟机

2.1 安装虚拟机 在本地搭建集群就需要这么几个事 装虚拟机 安装环境 配置集群 启动 这篇博客主要就是讲的装虚拟机这一个环节的 装虚拟机就是和组装一台现实中的电脑一样&#xff0c;首先来说就是要有硬件&#xff0c;就是组装硬件&#xff0c;然后就是装系统&#xff…

SpringCloud学习记录|day5

学习材料 2024最新SpringCloud微服务开发与实战&#xff0c;java黑马商城项目微服务实战开发&#xff08;涵盖MybatisPlus、Docker、MQ、ES、Redis高级等&#xff09; 目前讲过的 可惜&#xff0c;自己基本没理解。 雪崩问题 1.服务保护&#xff1a; 请求限流&#xff0c;线…

量化择时技术指标详解及实战应用(一)

🌟作者简介:热爱数据分析,学习Python、Stata、SPSS等统计语言的小高同学~🍊个人主页:小高要坚强的博客🍓当前专栏:《Python之量化交易》🍎本文内容:量化择时技术指标详解及实战应用(一)🌸作者“三要”格言:

从零开始使用最新版Paddle【PaddleOCR系列】——文本检测和识别模型的环境安装与基础使用

目录 一、环境安装配置 1.基本环境配置&#xff1a;torch与paddlepaddle安装 2.专精任务配置&#xff1a;PaddleX与PaddleOCR插件安装 3.测试数据配置&#xff1a;测试数据集下载与验证 二、模型基础使用 1.使用OCR模型预测 ​ 2.使用Detect检测模型 ​ 3.使用…

【论文笔记】Fine-tuned CLIP Models are Efficient Video Learners

&#x1f34e;个人主页&#xff1a;小嗷犬的个人主页 &#x1f34a;个人网站&#xff1a;小嗷犬的技术小站 &#x1f96d;个人信条&#xff1a;为天地立心&#xff0c;为生民立命&#xff0c;为往圣继绝学&#xff0c;为万世开太平。 基本信息 标题: Fine-tuned CLIP Models a…

QT实现改变窗口大小其子控件也自动调节大小

创建一个顶层布局即可&#xff0c;一定要在MainWindows或者Widget的下面&#xff01; 观察图标变化 带有禁止的意思是分拆布局&#xff08;当前无布局&#xff09; 现在是添加布局后了 注意&#xff1a;一定是在MainWindows或Widget才可以添加顶层布局&#xff0c;才可以实现…

antDesign Form.List下的Form.Item如何通过setFieldsValue设置值

翻了一下antDesign官网只看见了Form可以使用setFieldsValue设置值&#xff0c;却没找到Form.List使用setFieldsValue设置值。 于是研究了一下&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 我的解决方案是&#xff1a; 先设置为空数组, 再设置成…

241014-绿联UGOSPro-通过虚拟机访问主机的用户目录及文件夹

如图所示&#xff0c;两种方式&#xff1b; 方式1: 通过Files中的Other Locations 添加主机ip&#xff0c;随后输入主机的用户名及密码即可系统及文件加载可能需要一段时间&#xff0c;有点卡&#xff0c;加载完应该就可以点击访问了 方式2: 通过命令行直接ssh/sftp userna…

解锁C++继承的奥秘:从基础到精妙实践(下)

文章目录 前言&#x1f950;五、多继承&#xff0c;菱形继承和菱形虚拟继承&#x1f9c0;5.1 多继承&#x1f9c0;5.2 菱形继承&#x1f9c0;5.3 虚拟继承&#xff08;解决菱形继承问题&#xff09;5.3.1 虚拟继承的语法&#xff1a;5.3.2 虚拟继承示例&#xff1a; &#x1f9…

C语言 | Leetcode C语言题解之第479题最大回文数乘积

题目&#xff1a; 题解&#xff1a; int largestPalindrome(int n){if (n 1) {return 9;}int upper pow(10, n) - 1;for (int left upper;; --left) { // 枚举回文数的左半部分long p left;for (int x left; x > 0; x / 10) {p p * 10 x % 10; // 翻转左半部分到其自…

相机畸变模型

文章目录 概述相机畸变类型径向畸变切向畸变 畸变数学模型径向畸变模型切向畸变模型畸变数学模型总结 去畸变数学过程去畸变步骤 畸变测试结论参考 概述 相机畸变是图像处理和计算机视觉中的常见问题。由于透镜的物理特性&#xff0c;图像边缘的物体往往会呈现扭曲&#xff0c…

(34)FFT与信号频谱(双边谱)

文章目录 前言一、仿真代码二、仿真结果画图 前言 本文首先使用MATLAB生成一段余弦信号&#xff0c;然后对其进行FFT变换&#xff0c;给出了信号的双边幅度谱。 一、仿真代码 代码如下&#xff08;示例&#xff09;&#xff1a; %% 生成余弦波 % 指定信号的参数&#xff0c;…

k8s中的微服务

一、什么是微服务 用控制器来完成集群的工作负载&#xff0c;那么应用如何暴漏出去&#xff1f;需要通过微服务暴漏出去后才能被访问 Service是一组提供相同服务的Pod对外开放的接口。 借助Service&#xff0c;应用可以实现服务发现和负载均衡。 service默认只支持4层负载均…