全排列问题

日升时奋斗,日落时自省 

目录

1、全排列

2、全排列II

3、子集

4、组合


1、全排列

首先要了解全排列是怎么样的

例如:数组[1,2,3]的全排列(全排列就是不同顺序排列方式)

例子所有的排列方式如:[1,2,3],[1,3,2],[2,1,3],[2,3,1],[3,1,2],[3,2,1]六种

画个图来解释一下:

代码:

    List<List<Integer>> result;  //终态 结果List<Integer> path;  //每条路径的结果public List<List<Integer>> permute(int[] nums) {result=new ArrayList<>();path=new ArrayList<>();int n=nums.length;if(n==0){return result;}//标记位boolean[] vis=new boolean[n];//参数  数组 当前层数  数组个数 标记位bfs(nums,0,n,vis);return result;}public void bfs(int[] nums,int index,int len,boolean[] vis){//层数等于数组个数表示路径已经收集齐一次 进行终态添加if(index==len){result.add(new ArrayList<>(path));return ;}//每层都会进行整个数组的遍历for(int i=0;i<len;i++){//但是需要判断标记位 该数字是否被标记使用,这里需要是没有使用过的数字if(!vis[i]){//进来后就算是被使用了, 进行标记一下vis[i]=true;//同时添加到路径中path.add(nums[i]);//进行下一层 遍历bfs(nums,index+1,len,vis);//如果下一层进行失败 当前层的该数也没有用了,因为这条路走不通//标记位删除的标记vis[i]=false;//同时删除当前数位path.remove(path.size()-1);}}}

2、全排列II

题目来源:. - 力扣(LeetCode)

题目的主要目的就是让全排列不要重复,重复的就不要写出来了

首先我们想前后一样就不选了,但是这个是需要一个前置条件的,需要给数组排个序

去重条件就是 nums[i]!=nums[i-1]并且当前位置标记位为false(说明没有选过)为进入条件

条件代码:!check[i]&&nums[i]!=nums[i-1]    (但是这样是不行的)

注:i-1是会越界的

再次修改的条件代码:!check[i]&&(i==0||nums[i]!=nums[i-1])

注:其实还是不完全的,如果同一层的check的上一个(check[i-1])为true说明选过了,此时nums[i]==nums[i-1],我们选还是不选,当然是选

其实全排列的规律就是  1(1)2(2)2(3)   1(1)2(3)2(2)   其实这里就相当于我们选了(1)(2)(3) 这种情况

放弃了(1)(3)(2)这种类似情况

注:这里红括号表示层数

方法一:

代码:

    List<List<Integer>> ret;    //终态集List<Integer> path;         //结果集boolean[] check;public List<List<Integer>> permuteUnique(int[] nums) {ret=new ArrayList<>();path=new ArrayList<>();check=new boolean[nums.length]; //标记位Arrays.sort(nums); //排序 为去重做的准备dfs(nums,0);return ret;}public void dfs(int[] nums,int pos){//终态条件if(path.size()==nums.length){ret.add(new ArrayList<>(path));return ;}for(int i=0;i<nums.length;i++){//当前标记位==false  &&   (i不能越界||前后不相等||同层上一个为true)if(!check[i]&&(i==0||nums[i]!=nums[i-1]||check[i-1])){path.add(nums[i]);check[i]=true;dfs(nums,pos+1);path.remove(path.size()-1);check[i]=false;}}}

方法二:

如果觉得上述代码比较难以接受的话的,只要理解全排列就行,这里可以使用其他集合来处理去重问题例如使用Set来去重

    List<List<Integer>> ret;List<Integer> path;boolean[] check;public List<List<Integer>> permuteUnique(int[] nums) {ret=new ArrayList<>();path=new ArrayList<>();check=new boolean[nums.length];Arrays.sort(nums);dfs(nums,0);//创建set集合Set<List<Integer>> set=new HashSet<>();//添加后进行去重for(int i=0;i<ret.size();i++){set.add(ret.get(i));}//清除ret原来的值ret.clear();//重新将去重后的数据进行添加for(List<Integer> tmp:set){ret.add(tmp);}return ret;}public void dfs(int[] nums,int pos){if(path.size()==nums.length){ret.add(new ArrayList<>(path));return ;}for(int i=0;i<nums.length;i++){if(!check[i]){path.add(nums[i]);check[i]=true;dfs(nums,pos+1);path.remove(path.size()-1);check[i]=false;}}}

3、子集

题目来源:LCR 079. 子集 - 力扣(LeetCode)

子集就是我们在数学里理解的意思:

[1,2,3]的子集就是[],[1],[2],[3],[1,2],[1,3],[2,3],[1,2,3]

图示:

注:当前层到下一层的位置都会+1

代码:

    List<List<Integer>> ret;   //终态集List<Integer> path;       //结果集(路径集)public List<List<Integer>> subsets(int[] nums) {ret=new ArrayList<>();path=new ArrayList<>();dfs(nums,0);return ret;}private void dfs(int[] nums, int pos) {//子集  是  每次都会进行添加ret.add(new ArrayList<>(path));//当前层的位置 是上层位置+1   pos就是上层位置+1来的for(int i=pos;i<nums.length;i++){path.add(nums[i]);//递推条件就是数组 和 下层的下一个位置dfs(nums,i+1);path.remove(path.size()-1);}}

4、组合

题目来源:77. 组合 - 力扣(LeetCode)

子集的升级版(个人是这么理解的)

给定一个数n和一个组合个数k

n=4 ,k=2

其实就是让我们把[1,2,3,4]进行组合:

结果为:[1,2],[1,3],[1,4],[2,3],[2,4],[3,4] (其实是有点像子集的只是稍微加了点处理)

子集没有限定条件来添加结果集,这里添加了条件是 当结果集为k 的时候添加到终态集中

这里还是拿n=3,k=2再来举一个例子会更清楚些

其实就是让我们把[1,2,3]进行组合:

结果为:[1,2],[1,3],[2,3]

图示:

代码:

其实仔细看看也就是只有条件边了,其他都没有改变

    List<List<Integer>> ret;List<Integer> path;int num,key;  //设置为全局变量方便获取,不用通过传参来使用public List<List<Integer>> combine(int n, int k) {ret=new ArrayList<>();path=new ArrayList<>();num=n;key=k;dfs(1);return ret;}public void dfs(int pos){//限定结果集的个数为kif(path.size()==key){ret.add(new ArrayList<>(path));return;}for(int i=pos;i<=num;i++){path.add(i);dfs(i+1);path.remove(path.size()-1);}}

注:如果友友们还是感觉有点陌生我换个方法来看看

    List<List<Integer>> ret;List<Integer> path;int n,key;  //设置为全局变量方便获取,不用通过传参来使用public List<List<Integer>> combine(int[] nums, int k) {ret=new ArrayList<>();path=new ArrayList<>();key=k;dfs(nums,0);return ret;}public void dfs(int[] nums,int pos){//限定结果集的个数为kif(path.size()==key){ret.add(new ArrayList<>(path));return;}for(int i=pos;i<nums.length;i++){path.add(nums[i]);dfs(nums,i+1);path.remove(path.size()-1);}}

注:这个题目是稍微改动了一下,作为例题来看,是不是就和子集很相似,就是变了个条件

总结:知道全排列,子集和组合不一定是为了这几个题,这些方法有时候更像是枚举的形式出现,可以作为地基,稍作改善很多题就在一个小范围内进行解决 

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

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

相关文章

大话设计模式之享元模式

享元模式是一种结构型设计模式&#xff0c;旨在有效地支持大量细粒度的对象共享&#xff0c;从而减少内存消耗和提高性能。 在享元模式中&#xff0c;对象分为两种&#xff1a;内部状态&#xff08;Intrinsic State&#xff09;和外部状态&#xff08;Extrinsic State&#xf…

初级软件测试常见问题

1.JMeter &#xff08;1&#xff09;在http请求的时候&#xff0c;消息体数据中的数据需要用{}和“”标记起来&#xff0c;变量要用${}括起来。 &#xff08;2&#xff09;在响应断言的时候&#xff0c;要根据测试模式输出的内容来改变测试字段&#xff0c;假如输出错误可以把…

vscode 调试debug gdb vector string等STL容器,指定长度

主要展示2个调试信息&#xff1a; 1. 数组 *tr20&#xff0c;指tr数组的前20个元素 2.Vector *(int(*)[5])a ,指a容器前5个元素&#xff0c;也可以解决1的问题 二维数组 -exec p/d b也可以 附&#xff1a;命令参考 gdb 调试常用命令 - 红旗kernel - 博客园 (cnblogs.com) GD…

书生·浦语大模型开源体系(五)笔记

&#x1f497;&#x1f497;&#x1f497;欢迎来到我的博客&#xff0c;你将找到有关如何使用技术解决问题的文章&#xff0c;也会找到某个技术的学习路线。无论你是何种职业&#xff0c;我都希望我的博客对你有所帮助。最后不要忘记订阅我的博客以获取最新文章&#xff0c;也欢…

定时器、PWM定时器、UART串口通信

我要成为嵌入式高手之4月15日ARM第八天&#xff01;&#xff01; ———————————————————————————— 定时器 S3C2440A 有 5 个 16 位定时器。其中定时器 0、1、2 和 3 具有脉宽调制&#xff08;PWM&#xff09;功能。定时器 4 是一个无 输出引脚的内部…

部署项目的时候的一些错误

项目打jar包&#xff0c;找不到资源&#xff0c;连接不上数据库 项目打包后无法运行 直接在idea运行可以 解决方法&#xff1a;pom文件中增加&#xff08;配置文件如果是yml&#xff0c;写yml&#xff09; <resources><resource><directory>src/main/java&…

MySQL—MySQL架构

MySQL—MySQL架构 MySQL逻辑架构图如下&#xff1a; Connectors连接器:负责跟客户端建立连接&#xff1b;Management Serveices & Utilities系统管理和控制工具&#xff1b;Connection Pool连接池:管理用户连接&#xff0c;监听并接收连接的请求&#xff0c;转发所有连接的…

使用Scrapy选择器提取豆瓣电影信息,并用正则表达式从介绍详情中获取指定信息

本文同步更新于博主个人博客&#xff1a;blog.buzzchat.top 一、Scrapy框架 1. 介绍 在当今数字化的时代&#xff0c;数据是一种宝贵的资源&#xff0c;而网络爬虫&#xff08;Web Scraping&#xff09;则是获取网络数据的重要工具之一。而在 Python 生态系统中&#xff0c;S…

深度学习遇到的疑问记录

深度学习遇到的疑问记录 1. Softmax和Sigmoid之间的区别和联系2. 语音唤醒模型为什么经常选择使用CNN而不是RNN3. 什么是前馈神经网络4. 前馈神经网络包含cnn吗4. 简单介绍一些LSTM网络结构5. 什么是GRU 1. Softmax和Sigmoid之间的区别和联系 Softmax和Sigmoid都是神经网络中常…

hadoop编程之部门工资求和

数据集展示 7369SMITHCLERK79021980/12/17800207499ALLENSALESMAN76981981/2/201600300307521WARDSALESMAN76981981/2/221250500307566JONESMANAGER78391981/4/22975207654MARTINSALESMAN76981981/9/2812501400307698BLAKEMANAGER78391981/5/12850307782CLARKMANAGER78391981/…

Rust语言入门第五篇-数据类型

文章目录 数据类型1.标量类型1. 整数类型2.浮点数类型f32 和 f64示例代码注意事项 3.布尔类型4.字符类型 2.复合类型整数类型技术细节1. 检查溢出&#xff08;Checking Overflow&#xff09;2. 溢出时 panic&#xff08;Panic on Overflow&#xff09;3. 使用 Wrapping 模式&am…

hadoop编程之工资序列化排序

数据集展示 7369SMITHCLERK79021980/12/17800207499ALLENSALESMAN76981981/2/201600300307521WARDSALESMAN76981981/2/221250500307566JONESMANAGER78391981/4/22975207654MARTINSALESMAN76981981/9/2812501400307698BLAKEMANAGER78391981/5/12850307782CLARKMANAGER78391981/…

【C语言基础】:预处理详解(二)

文章目录 一、宏和函数的对比二、#和##运算符2.1 #运算符2.2 ##运算符 三、#undef四、命令行定义五、条件编译六、头文件的包含1. 头文件包含的方式2. 嵌套文件包含 上期回顾&#xff1a; 【C语言基础】&#xff1a;预处理详解(一) 一、宏和函数的对比 宏通常被应有于执行简单…

蓝桥杯-回文日期

回文日期&#xff1a;回文日期 //找到一个年份&#xff0c;倒序输出符合条件就行 #include<iostream> using namespace std; string tmp1,tmp2; int mx0; int flag11,flag20; bool is(int n){int date[20]{0,31,28,31,30,31,30,31,31,30,31,30,31};int dayn%100;int m…

【Quartz】Quartz表的SQL语句,版本2.3.0

下载地址 https://www.quartz-scheduler.org/downloads/files/quartz-2.3.0-distribution.tar.gz 文件所在路径 quartz-2.3.0-SNAPSHOT\src\org\quartz\impl\jdbcjobstore 一般使用MySQL的场景下关注tables_mysql.sql和tables_mysql_innodb.sql两个文件即可&#xff0c;笔者…

Web前端-JavaScript

黑马程序员JavaWeb开发教程 文章目录 一、js引入方式1、内部脚本2、外部脚本 二、js基础语法1、书写语法&#xff08;1&#xff09;基本语法&#xff08;2&#xff09;输出语句 2、变量&#xff08;1&#xff09;变量&#xff08;2&#xff09;注意事项 3、数据类型、运算符、流…

腾讯云服务器CVM标准型S8实例CPU内存、网络和存储性能测评

腾讯云第八代云服务器标准型S8实例基于全新优化虚拟化平台&#xff0c;CPU采用Intel Emerald Rapids 全新处理器&#xff0c;睿频3.0GHz&#xff0c;内存采用最新DDR5&#xff0c;默认网络优化&#xff0c;最高内网收发能力达4500万pps&#xff0c;最高内网带宽可支持120Gbps。…

Docker搭建LibreSpeed

LibreSpeed 是一个轻量级的网络速度测试工具&#xff0c;它使用 JavaScript 编写&#xff0c;通过 XMLHttpRequest 和 Web Workers 进行数据传输&#xff0c;无需 Flash、Java 或 WebSocket 支持。LibreSpeed 提供了类似于 Speedtest by Ookla 的本地部署解决方案&#xff0c;可…

人工智能2024年发展方向

预计到2024年&#xff0c;人工智能领域将继续迅速发展&#xff0c;主要的发展方向可能包括&#xff1a; 强化学习的进一步应用&#xff1a;强化学习在自动驾驶、游戏、金融等领域已经取得了一定成就&#xff0c;未来将继续深入研究和应用&#xff0c;包括在更复杂的环境中的实际…

java编译过程

java编译器将 java 源文件转换成 class 文件的过程。 &#xff08;1&#xff09;词法分析器 作用&#xff1a;将Java源文件的字符流转变成对应的Token流 每个词法单元&#xff08;token&#xff09;都有一个类型&#xff08;token type&#xff09;和一个值&#xff08;toke…