穷举vs暴搜vs深搜vs回溯vs剪枝(一)
文章目录
- 全排列
- 子集
- 找出所有子集的异或总和再求和
- 全排列 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] = true和check[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();}}
};
相关文章:
穷举vs暴搜vs深搜vs回溯vs剪枝(一)
文章目录 全排列子集找出所有子集的异或总和再求和全排列 II电话号码的字母组合 全排列 题目:全排列 思路 通过深度优先搜索的方式,不断枚举每个数在当前位置的可能性,然后回溯到上一个状态,直到枚举完所有可能性得到正确的结果 r…...
枚举的应用
1.枚举的语法特点 枚举是jdk1.5提供的一个特性 枚举是一个特殊的类,这个类的对象的数量是有限的。在定义枚举类的同时就已经确定了类对象及类对象的数量。 枚举使用enum关键字定义 class A{} enum A{} 在枚举类中的第一行,就需要提供枚举类的对象&a…...
读数据工程之道:设计和构建健壮的数据系统14源系统
1. 源系统中的数据生成 1.1. 数据工程师的工作是从源系统获取数据,对其进行处理,使其有助于为下游用例提供服务 1.2. 数据工程师的角色将在很大程度上转向理解数据源和目的地之间的相互作用 1.3. 数据工程的最基本的数据管道任务——将数据从A移动到B…...
基于SpringBoot+Vue的厨艺交流系统的设计与实现(源码+定制开发)厨艺知识与美食交流系统开发、在线厨艺分享与交流平台开发、智能厨艺交流与分享系统开发
博主介绍: ✌我是阿龙,一名专注于Java技术领域的程序员,全网拥有10W粉丝。作为CSDN特邀作者、博客专家、新星计划导师,我在计算机毕业设计开发方面积累了丰富的经验。同时,我也是掘金、华为云、阿里云、InfoQ等平台…...
STMicroelectronics 意法半导体芯片选型表
意法半导体作为全球知名的半导体厂商,其产品广泛应用于各个领域,从消费电子到工业控制,从汽车电子到通信设备,都能看到意法半导体芯片的身影。在电子硬件设计领域,芯片的选型至关重要。亿配芯城(ICgoodFind…...
TCP/IP 寻址
TCP/IP 寻址 概述 TCP/IP(传输控制协议/互联网协议)是一组用于数据网络的通信协议。它们定义了数据如何在网络上从一个设备传输到另一个设备。在TCP/IP网络中,每个设备都有一个唯一的地址,称为IP地址,用于标识网络上…...
深入探索 APKTool:Android 应用的反编译与重打包工具
文章目录 一、反编译 APK1.1 解压 APK1.2 DEX 文件转换1.3 资源解码 二、重新打包 APK2.1 资源重新编译2.2 smali 转换为 DEX2.3 打包 APK2.4 签名 APK 三、技术原理3.1 Smali/Baksmali3.1.1 DEX 文件格式3.1.2 Smali 语法3.1.2.1 指令3.1.2.2 寄存器3.1.2.3 操作码3.1.2.4 注释…...
软件测试与软件缺陷的基础知识
✨✨ 欢迎大家来访Srlua的博文(づ ̄3 ̄)づ╭❤~✨✨ 🌟🌟 欢迎各位亲爱的读者,感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢,在这里我会分享我的知识和经验。&am…...
【JPCS独立出版,EI检索稳定】第三届能源互联网及电力系统国际学术会议(ICEIPS 2024)
第三届能源互联网及电力系统国际学术会议(ICEIPS 2024) 2024 3rd International Conference on Energy Internet and Power Systems ICEIPS 2024已成功申请JPCS - Journal of Physics: Conference Series (ISSN:1742-6596) ICEIPS 2024独立出版&…...
ssm配置模式
新版 用Java类,全注解demo案例 1. AppConfig.java (Spring主配置类)package com.example.config;import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.cont…...
[MySQL课后作业]人事管理系统的SQL实践
第一题 1.假设某商业集团中有若干公司,人事数据库中有3个基本表: 职工表:EMP(E#.ENAME,AGE, SEX, ECITY)。 其属性分别表示职工工号、姓名、年龄、性别和居住城市。 工作表:WORKS(E#,C#,SALARY)。其属性分别表示职工工号、所在公司的编号和工资。 公司表:COMP(C#,CA…...
【MySQL】增删改查-进阶(二)
目录 🌴新增 🎄查询 🚩聚合查询 🏀聚合函数 🏀group by子句 🏀HAVING 🚩联合查询 🏀内连接 🏀外连接 🏀自连接 🏀子查询 🏀…...
cefsharp79.1.360(Chromium 79.0.3945.130)支持H264视频播放-PDF预览 老版本回顾系列体验
一、关于此版本 版本:Cef 79.1.36/CefSharp 79.1.360/Chromium 79.0.3945.130/支持H264/支持PDF预览 支持PDF预览和H264推荐版本 63/79/84/88/100/111/125 运行环境需要 visual c++ 2015不支持xp/vista/2003/2008默认不支持h264(版权问题)支持打印预览 print preview已知问题…...
【Linux】main函数的参数列表从何而来?
Linux系统进程通过exec系列函数启动新程序时,argc整型 、 argv数组 和 环境变量表 environ 会作为 exec 系列函数的参数,显式传递给新程序的 main 函数。 main函数的参数列表 在C语言中,main函数的标准参数列表通常如下所示: in…...
缓冲区类QBuffer
1、QBuffer继承自QIODevice 2、是一种随机设备 3、和QFile类似, 4、有了 QBuffer,你可以把 QByteArray 当成文件一样来操作 其主要作用就是像QFile操作文件一样来操作一块QByteArray(内存区域),比如读和写 常用方…...
从一个事故中理解 Redis(几乎)所有知识点
作者:看破 一、简单回顾 事故回溯总结一句话: (1)因为大 KEY 调用量,随着白天自然流量趋势增长而增长,最终在业务高峰最高点期占满带宽使用 100%。   (2ÿ…...
MySQL程序介绍<二>
目录 mysqlcheck - 表维护程序 Mysqldump - 数据库备份程序 mysqladmin - MySQL 服务器管理程序 mysqlshow - 显⽰数据库、表和列信息 mysqldumpslow - 总结慢查询⽇志⽂件 编辑 mysqlbinlog - 处理⼆进制⽇志⽂件 mysqlslap - 负载仿真客⼾端 接着上篇继续介绍MySQL…...
Java项目实战II基于Spring Boot的毕业就业信息管理系统设计与实现(源码+数据库+文档)
目录 一、前言 二、技术介绍 三、系统实现 四、文档参考 五、核心代码 六、源码获取 全栈码农以及毕业设计实战开发,CSDN平台Java领域新星创作者,专注于大学生项目实战开发、讲解和毕业答疑辅导。获取源码联系方式请查看文末 一、前言 随着高校扩…...
LeetCode 1343.大小为K且平均值大于等于阈值的子数组数目
题目: 给你一个整数数组 arr 和两个整数 k 和 threshold 。 请你返回长度为 k 且平均值大于等于 threshold 的子数组数目。 思路:定长滑动窗口 入 更新 出 代码: class Solution {public int numOfSubarrays(int[] arr, int k, int t…...
【电商项目】1分布式基础篇
1 项目简介 1.2 项目架构图 1.2.1 项目微服务架构图 1.2.2 微服务划分图 2 分布式基础概念 3 Linux系统环境搭建 查看网络IP和网关 linux网络环境配置 补充P123(修改linux网络设置&开启root密码访问) 设置主机名和hosts映射 主机名解析过程分析&…...
告别在线转换!用PowerShell+FFmpeg批量把FLAC无损转成ALAC(附完整脚本)
打造高效音频工作流:PowerShellFFmpeg批量转换FLAC到ALAC全攻略 每次整理音乐库时,最头疼的就是格式兼容性问题。上周我帮朋友迁移他的2000多首FLAC音乐到苹果设备,原本打算用在线转换工具,结果光是上传就花了整整一天——这还不算…...
从PHY芯片看工业网络精准时钟:IEEE 1588v2(PTP)协议实现与选型指南
1. 工业网络为何需要纳秒级时钟同步? 在工业自动化生产线或通信基站里,你可能见过这样的场景:几十台机械臂协同装配零件时,某个关节动作偏差1毫秒,整个产品就可能报废;5G基站切换时,时间误差超过…...
OpenAlternative移动端优化完全指南:打造完美开源软件目录响应式体验
OpenAlternative移动端优化完全指南:打造完美开源软件目录响应式体验 【免费下载链接】openalternative Curated list of open source alternatives to proprietary software. 项目地址: https://gitcode.com/gh_mirrors/op/openalternative 在移动设备使用率…...
HelloWord-Keyboard固件编程完全指南:从零掌握机械键盘定制开发
HelloWord-Keyboard固件编程完全指南:从零掌握机械键盘定制开发 【免费下载链接】HelloWord-Keyboard 项目地址: https://gitcode.com/gh_mirrors/he/HelloWord-Keyboard 想要打造属于自己的智能机械键盘吗?HelloWord-Keyboard项目为你提供了一个…...
OpenClaw隐私保护方案:Qwen3-14B本地处理VS第三方API对比
OpenClaw隐私保护方案:Qwen3-14B本地处理VS第三方API对比 1. 隐私保护的核心战场 去年帮朋友处理一个自动化需求时,我第一次意识到AI助手的隐私边界问题。他们团队需要处理大量客户访谈录音,但使用某知名云端AI服务后,法务部门突…...
tmux 示例
技术文章大纲示例:人工智能在医疗诊断中的应用 引言 概述人工智能在医疗领域的重要性当前医疗诊断面临的挑战人工智能技术的引入如何改变传统诊断方式 人工智能技术基础 机器学习与深度学习的核心概念计算机视觉在医疗影像分析中的作用自然语言处理(NLP&…...
.Net基于AgentFramework中智能体Agent Skill集成Shell命令实现小龙虾mini版峡
从0构建WAV文件:读懂计算机文件的本质 虽然接触计算机有一段时间了,但是我的视野一直局限于一个较小的范围之内,往往只能看到于算法竞赛相关的内容,计算机各种文件在我看来十分复杂,认为构建他们并能达到目的是一件困难…...
支持立式卧式插板继电器输入3-40V控制,5-10mA电流,250V AC 电流3-8A
替代原装 AQG22105 AQG22112 AQG22124 AQG22224 AQG22205 AQG22212 G3MC-202PL-VD-12V 东芝的TS21j48S、TSA3100J;厦门宏发的JGC-4F-12D-1M;三菱的SWIDD-H1-4C;欧姆龙的G3MC-202PL-VD-2;三菱SW2DE-H1-4等...
GCC优化禁用指南:精准控制编译行为的5种方法
1. 为什么需要禁用GCC优化? 在嵌入式开发或者调试过程中,我们经常会遇到一些奇怪的bug:明明代码逻辑没有问题,但程序运行时却出现异常。这时候很可能就是编译器优化在"捣鬼"。GCC作为最常用的开源编译器,它的…...
RIT库:ARM Cortex-M高精度周期性中断定时器实现
1. RIT库概述:嵌入式系统中的高精度周期性中断定时器实现RIT(Repetitive Interrupt Timer)库是一个专为ARM Cortex-M系列微控制器设计的轻量级、高精度周期性中断定时器抽象层。其核心目标并非替代硬件外设本身,而是提供一套统一、…...
