【递归、搜索与回溯】综合练习一

综合练习一

  • 1.找出所有子集的异或总和再求和
  • 2.全排列 II
  • 3.电话号码的字母组合
  • 4.括号生成

在这里插入图片描述

点赞👍👍收藏🌟🌟关注💖💖
你的支持是对我最大的鼓励,我们一起努力吧!😃😃

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

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

题目描述:

在这里插入图片描述

先找出所有子集,然后把每个子集异或的和加起来返回去。
在这里插入图片描述

算法原理:
这道题和我们上一道思路完全是一模一样,

  1. 先画出决策树
  2. 设计代码
    全局变量
    递归函数
    细节:回溯、剪枝、递归出口

因为我们有上一道题的基础,我们直接就画出决策树
在这里插入图片描述
这里我们需要两个全局变量,一个path记录到沿途每个子集的异或,然后sum负责每个字节的异或和加起来。dfs函数,还是需要一个pos记录当前异或的位置,dfs(nums,pos),回溯利用异或的规则,两个相同的数异或为0。这里也没有剪枝, 递归出口 循环结束就是递归出口。

class Solution {int sum=0;int path=0;
public:int subsetXORSum(vector<int>& nums) {dfs(nums,0);return sum;}void dfs(vector<int>& nums,int pos){sum+=path;for(int i=pos;i<nums.size();++i){path^=nums[i];dfs(nums,i+1);path^=nums[i]; // 恢复现场}}
};

2.全排列 II

题目链接:47. 全排列 II

题目分析:

在这里插入图片描述

重复的数全排列后会有重复的结果,这道题就是要求去掉重乎之后的全排列

在这里插入图片描述

算法原理:
这道题几乎和全排列1 一模一样,我们就不在细说那些决策树怎么画,代码应该怎么写等等。这里主要就是剪枝的问题。

下面我们边画决策树变分析问题,把全排列所有不重复不漏的情况画出来,越详细越好。 我们只用关心四个位置,每个位置每次从数组中4个数选择一个树放到一个位置上就行了。只用选四次就行了。

第一次可以选第一个1、第二个1、第三个1、2,但是注意这里就存在剪枝的问题了,如果第一个位置还把第二个1和第三个1选上,此时就会存在重复问题!因为后面三个位置是从112中选的。
在这里插入图片描述
此时就出现了第一种剪枝情况

同一个节点的所有分支中,相同的元素只能选择一次

在这里插入图片描述

然后我们再往下走,第二个位置也可以从数组中4个数字中选任意一个。但是第一个1我们要把它剪掉,因为第一个位置已经把第一个1选过了,只能选一次。此时就有了第二种剪枝情况,这个是和全排列1一模一样的。

同一个位置的数,只能使用一次
还是用bool类型的check数组可以实现,check[i] = true 表明已经使用过了,
check[i] = false 说明还没有使用过。
在这里插入图片描述

但是这里还会出现剪枝情况,第一个1不能用,那我可以把第二个1放在第二个位置,但是第三个1不能出现了,因为同一个节点分支相同的数只能出现一次!

在这里插入图片描述
接下来我们就不画了,我们就可以写代码了。代码逻辑和全排列1几乎一模一样,这里我们主要分析,剪枝应该怎么写。剪枝可以从两个角度写。其实就是两个if判定语句。
1.只关心 “不合法” 的分支。 不合法的直接不让递归下去
2.只关心 “合法” 的分支 合法的就递归
虽然是两个角度,但是最终得到结果都是一样的。

只关心 “不合法” 的分支

  1. 当有一个位置已经选了这个数了下一个位置就不能在选这个位置,check[i] == true
  2. 属于同一个分支节点,前面的数和当前的数相等 就不能选当前的数了。nums[i] == nums[i-1],但是这里有一个问题,我们这个数组里面的数字是有序的,可以这样写,如果无序就不能这样写,因此最开始先对数组进行排序。但是还是有问题,目前nums[i] == nums[i-1]这个条件太广泛了,注意看它目前只适用于第一个位置中不选择相同数,但是在第二个位置中又出现第二个1可以选第三个1不选的情况。因此还要再加条件,注意我们是从第一个位置递归下去到第二个位置然后出现相同的位置不选,但是当我们递归返回的时候这个数又可以选了。这个check[i]==true是上一层的和当前属于第二层无关!我们关注的是同一层相同元素只能选一次。因此这个条件是
    nums[i] == nums[i-1] && check[i-1] == false,但是还不够,可能会有越界的风险,因此这个条件最终是 i != 0 && nums[i] == nums[i-1] && check[i-1] == false

在这里插入图片描述
把上面两个条件组合在一起就得到只关心 “不合法” 的分支
在这里插入图片描述

只关心 “合法” 的分支

  1. 当一个数没人选的时候可以选这个数 check[i] == false
  2. 但注意到同一层可能有相同的数,第一个相同的数没人选因为是第一次出现确实是可以选的,但是如果当前的数和前面的数相同即使当前数没人选也是不能选的,因此 还要满足 nums[i] != num[i-1] 只要这个数不和前面相同说明就是第一次出现了绝对可以选!。但是还有一种情况,如果有相同的数字,也就是 nums[i] != num[i-1] 不满足的情况,那就是满足 nums[i] == num[i-1] 前面数和后面数相同的情况,如果上一个位置选了前面的数,那走下到下一个位置的时候,因为这个数已经选过了check[i] == true,其实也就说明这个数是上一个位置的数,和我本次第二个位置选数无关,即使是相同的数子也没有关系,我也是能够选择这个后面相同的数。也就是要满足 check[i-1] == true 前面的相同数被上一个位置选了。还有最后一个情况,刚开始i==0肯定是第一次出现的数并且这个数字没人选的时候,可以选。

在这里插入图片描述

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);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){// 剪枝// 1.只关心不合法// 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);// check[i]=false;// path.pop_back();//恢复现场// 2.只关心合法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);check[i]=false;path.pop_back();//恢复现场}}}
};

3.电话号码的字母组合

题目链接:17. 电话号码的字母组合

题目分析:

在这里插入图片描述
就是数字对应的字符串进行排列组合。对于这种搜索啊、暴搜啊,我们已经知道要用到递归,回溯、剪枝了。
在这里插入图片描述
算法原理:
有了前面的基础,这个题我们就不写那么步骤了,画出决策树,我们直接写出对应需要的东西。不过在此之前我们需要先将数字与字符串映射关系搞和,我们可以用哈希表映射,或者其他方法,这里最简单的就是弄一个字符串数组把数字和字符串映射一下。

接下来就是画出决策树然后分析一下,首先需要两个全局变量 ret记录结果,path记录每条路径到叶子节点的组合,递归函数 给我一个digits和数字然后递归函数把组合后的结果返回出来,相信它一定能完成任务。dfs(digits,pos),然后回溯 记得恢复现场,递归出口 到叶子节点,这道题没有剪枝

在这里插入图片描述

class Solution {
public:string numberletter[10]={"","","abc","def","ghi","jkl","mno","pqrs","tuv","wxyz"};vector<string> ret;string path;vector<string> letterCombinations(string digits) {if(digits.empty()) return ret;dfs(digits,0);return ret;}void dfs(string& digits,int pos){if(pos == digits.size()){ret.push_back(path);return;}string str=numberletter[digits[pos]-'0'];for(int i=0;i<str.size();++i){path+=str[i];dfs(digits,pos+1);path.pop_back();}}
};

4.括号生成

题目链接:22. 括号生成

题目分析:

在这里插入图片描述
给几对括号,然后把括号组合一下形成 有效括号
在这里插入图片描述
算法原理:
首先我们要知道什么是 有效括号的组合
1.左括号的数量 = 右括号的数量
2.从头开始的任意一个子串,左括号数量 >= 右括号数量

如下面第二种情况,虽然满足条件1但是并不满足条件2,画线字串右括号数量 > 左括号数量
在这里插入图片描述

对于这样暴力枚举的所有情况的问题,我们还是画一颗决策树,把所有情况不重不漏的情况都画出来,然后根据这棵树我们写代码。

每个位置都有两种选择,但是注意到刚开始就有剪枝的情况,刚开始不能选右括号,因为不满足条件2,所以右括号我们要分情况剪枝。当右括号数量大于等于左括号的数量时此时不能添加右括号 right >= left。还有当左括号的数量大于等于n时此时就没有左括号可以选了,left >= n。后面情况都是这样分析的,因此我们就可以做写代码的准备了。
在这里插入图片描述
全局变量,需要一个 left 记录左括号的数量,right 记录右括号的数量,还有一个n记录有几对括号要组合。还需要一个ret记录结果,path记录每条路径的结果。递归函数 每个位置都有两种选择。因为上面都是用的全局变量,因此递归函数参数什么都不用传了。dfs()。回溯 当把path放到ret里,返回后要恢复现场。pop掉path最后一个位置元素。剪枝 前面已经分析了。递归出口 当right==n的时候说明括号组合完了。

class Solution {
public:int left=0,right=0;vector<string> ret;string path;vector<string> generateParenthesis(int n) {dfs(n);return ret;}void dfs(int& n){if(right == n){ret.push_back(path);return;}if(left < n) //添加左括号{path+='(';++left;dfs(n);path.pop_back();--left;//恢复现场}if(right<left)//添加右括号{path+=')';++right;dfs(n);path.pop_back();--right;//恢复现场}}
};

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

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

相关文章

【stable diffusion】ComfyUI扩展安装以及”127.0.0.1拒绝了我们的连接请求“解决记录

扩展安装 虽然大家都推荐将扩展包直接放到extension文件夹的方式&#xff0c;但我还是推荐直接在sd webui的扩展处下载&#xff0c;酱紫比较好维护一点&#xff0c;我个人感觉。 按照上图顺序点击会出现”URLError: <urlopen error [Errno 11004] getaddrinfo failed>”…

2024 Java 异常—面试常见问题

目录 一、异常的分类 二、throw和throws都是异常处理的关键字&#xff0c;二者区别。 三、try-catch-finally 中&#xff0c;如果 catch 中 return 了&#xff0c;finally 还会执行吗&#xff1f; 四、try-catch-finally 中哪个部分可以省略&#xff1f; 五、常见的 Runti…

BarTender软件下载附加详细安装教程

BarTender是美国海鸥科技推出的一款优秀的条码打印软件&#xff0c;应用于 WINDOWS95 、 98 、 NT 、 XP 、 2000 、 2003 和 3.1 版本&#xff0c; 产品支持广泛的条形码码制和条形码打印机&#xff0c; 不但支持条形码打印机而且支持激光打印机&#xff0c;还为世界知名品牌条…

C脚本实现用键盘按键控制Wincc某按钮动作

文章目录 前言一、创建Wincc画面并添加变量及按钮二、在“事件”-“键盘”下&#xff0c;编写“按下”和“释放”的C脚本 前言 在某些特定场景下&#xff0c;需要通过电脑键盘控制上位机界面上按钮按下或释放&#xff0c;本文给出了基于C脚本的解决方案。 一、创建Wincc画面并…

Tailwind CSS 响应式设计实战指南

title: Tailwind CSS 响应式设计实战指南 date: 2024/6/13 updated: 2024/6/13 author: cmdragon excerpt: 这篇文章介绍了如何运用Tailwind CSS框架创建响应式网页设计&#xff0c;涵盖博客、电商网站及企业官网的布局实例&#xff0c;包括头部导航、内容区域、侧边栏、页脚…

从零开始手把手Vue3+TypeScript+ElementPlus管理后台项目实战十一(整体布局04之Header及用户注销)

新增Hearder 新增 src/layout/components/PageHeader.vue <template><div class"header-cont"><div><h1><router-link to"/">RealWorld</router-link></h1></div><div><template v-if"is…

北斗三号短报文终端户外应急通信解决方案

北斗三号短报文终端户外应急通信解决方案主要基于我国自主研发的北斗三号全球卫星导航系统&#xff08;BDS-3&#xff09;&#xff0c;为户外应急场景提供高效、稳定的通信服务。以下是对该解决方案的详细阐述&#xff1a; 一、方案概述 北斗三号短报文终端户外应急通信解决方…

MapperStruct拷贝数据的介绍和使用

1、前言 在java 编程中&#xff0c;对象直接拷贝是很常用的方法&#xff0c;最初我们常用spring提供的拷贝工具BeanUtils的copyProperties方法完成对象之间属性的拷贝。但是它有几个明显的如下缺点 1、属性类型不一致导致摸一个属性值拷贝失败 2、通一个字段使用基本类型和包…

数据结构之链表的经典笔试题

找往期文章包括但不限于本期文章中不懂的知识点&#xff1a; 个人主页&#xff1a;我要学编程(ಥ_ಥ)-CSDN博客 所属专栏&#xff1a;数据结构&#xff08;Java版&#xff09; 目录 203. 移除链表元素 206. 反转链表 876. 链表的中间节点 面试题 02.02. 返回倒数第k个节点 …

【AI】文心一言的使用分享

在数字化时代&#xff0c;人工智能&#xff08;AI&#xff09;技术的飞速发展正在改变我们的生活。文心一言&#xff0c;作为这一浪潮中的佼佼者&#xff0c;以其卓越的自然语言处理能力和广泛的应用场景&#xff0c;给我带来了前所未有的使用体验。在这篇分享中&#xff0c;我…

Spring Cloud Bus 消息总线基础入门与实践总结

【1】基础介绍 其主要是实现分布式自动刷新配置功能&#xff0c;Spring Cloud Bus 配合 Spring Cloud Config 使用可以实现配置的动态刷新。Spring Cloud Bus是用来将分布式系统的节点与轻量级消息系统链接起来的框架&#xff0c;它整合了Java的事件处理机制和消息中间件的功能…

Wall国内开源程序照片墙,支持VR全景及安装教程

下载 GitHub官网&#xff1a;https://github.com/zhang-tong-yao/wall 软件下载&#xff1a;https://github.com/zhang-tong-yao/wall/releases&#xff0c;推荐下载最新的版本。 演示效果 目前支持PC端和移动端自适应。 演示地址&#xff1a;https://demo-wall.ityao.cn …

农产品商城系统农资电商商城系统java项目jsp web项目

农产品商城系统农资电商商城系统java项目jsp web项目 会员用户可以实现&#xff1a;注册登录、商品信息浏览、加入购物车、加入收藏、下单支付、查看公告、查看以及留言、个人信息管理 后台管理的实现&#xff1a;类别管理模块、商品管理模块、会员管理模块、订单管理、退货管…

你项目用到了 Dubbo,讲讲为什么要用以及怎么使用的?

引言&#xff1a;在现代分布式系统中&#xff0c;服务化架构已成为构建高性能、可扩展和可靠系统的关键。 Dubbo 作为一种优秀的服务化框架&#xff0c;提供了丰富的特性和灵活的配置选项&#xff0c;使得它在业界得到了广泛的应用和认可。本文将探讨为什么选择 Dubbo 以及如何…

JavaEE大作业之班级通讯录系统(前端HTML+后端JavaEE实现)PS:也可选网络留言板、图书借阅系统、寝室管理系统

背景&#xff1a; 题目要求&#xff1a; 题目一&#xff1a;班级通讯录【我们选这个】 实现一个B/S结构的电子通讯录&#xff0c;其中的每条记录至少包含学号、姓名、性别、班级、手机号、QQ号、微信号&#xff0c;需要实现如下功能&#xff1a; &#xff08;1&#xff09;…

Objective-C基础语言开发来袭,你准备好了吗?

文/ZaiZai 前言 今天小白电脑技术的公众号迎来了一位Objective-C语言开发大神——ZaiZai。接下来有想要学习写插件&#xff08;iOS/macOS/iPadOS/tvOS&#xff09;的小伙伴可以关注微信公众号&#xff0c;教程将持续更新。 ZaiZai个人介绍……呃……他不让放。 Objective-C…

AI早班车6.13

全球AI新闻速递 1.Yandex 开源模型训练工具“YaFSDP”&#xff0c;提升 GPU 利用率。 2.全天候激光智能除草机器人落地&#xff0c;环境零污染。 3.苹果2024开发者大会-Siri 更智能。 4.CampusAI筹集1000万美元种子资金&#xff0c;创建虚拟校园用于学习AI技能。 5.OpenAI&…

C语言详解(预编译)

Hi~&#xff01;这里是奋斗的小羊&#xff0c;很荣幸您能阅读我的文章&#xff0c;诚请评论指点&#xff0c;欢迎欢迎 ~~ &#x1f4a5;&#x1f4a5;个人主页&#xff1a;奋斗的小羊 &#x1f4a5;&#x1f4a5;所属专栏&#xff1a;C语言 &#x1f680;本系列文章为个人学习…

什么是DMZ?路由器上如何使用DMZ?

文章目录 📖 介绍 📖🏡 演示环境 🏡📒 DMZ 📒🚀 DMZ的应用场景💡 路由器设置DMZ🎈 注意事项 🎈⚓️ 相关链接 ⚓️📖 介绍 📖 在网络管理中,DMZ(Demilitarized Zone,隔离区)是一个特殊的网络区域,常用于将公共访问和内部网络隔离开来。DMZ功能允许…

遇到JSON文件就头大?掌握Python这几种方法,让你轻松应对

目录 1、标准库json模块 📄 1.1 json.load()函数介绍 1.2 json.loads()处理字符串 1.3 使用json.dump()写入JSON 1.4 json.dumps()美化输出 1.4 错误处理与编码问题 1.5 高效读取大文件技巧 2、第三方库simplejson加持 🔧 2.1 安装与导入simplejson 2.2 性能优势与…