031.下一个排列Java实现

题意


 

整数数组的一个 排列 就是将其所有成员以序列或线性顺序排列。


 

例如,arr = [1,2,3] ,以下这些都可以视作 arr 的排列:[1,2,3]、[1,3,2]、[3,1,2]、[2,3,1] 。


 

整数数组的 下一个排列 是指其整数的下一个字典序更大的排列。更正式地,如果数组的所有排列根据其字典顺序从小到大排列在一个容器中,那么数组的 下一个排列 就是在这个有序容器中排在它后面的那个排列。


 

如果不存在下一个更大的排列,那么这个数组必须重排为字典序最小的排列(即,其元素按升序排列)。


 

  1. 例如,arr = [1,2,3] 的下一个排列是 [1,3,2] 。
  2. 类似地,arr = [2,3,1] 的下一个排列是 [3,1,2] 。
  3. 而 arr = [3,2,1] 的下一个排列是 [1,2,3] ,因为 [3,2,1] 不存在一个字典序更大的排列。


 

给你一个整数数组 nums ,找出 nums 的下一个排列。


 

必须 原地 修改,只允许使用额外常数空间。


 

难度


 

中等


 

示例

例1


 

输入:nums=[1,2,3]

输出:[1,3,2]

输入:nums=[3,2,1]

输出:[1,2,3]

输入:nums=[1,1,5]

输出:[1,5,1]


 

分析 1


 

这道题看起来很唬人,比如题目描述中提到的“字典序”,很多人第一眼看到这个名词的时候会有点懵,这里简单解释一下。


 

字典序(dictionary order),又称 字母序(alphabetical order),原意是表示英文单词在字典中的先后顺序,在计算机领域中扩展成两个任意字符串的大小关系。


 

在这道题目中,字典序其实是指☞ 数字大小的先后顺序


 

理解了“字典序”,就能很快理解这道题的题意了:要我们求出比这个数更大的一个排列,比如 123,比它大的是 132,对吧?只需要改变 23 的位置就可以了。312 也比 123 大,只不过它是比 132 更大的一个,不是比 123 更大的一个,官大一级压死人,一级一级来压(😂)。


就好像我们在打扑克牌,我出了一个对 10,那你出对 11 就压住我了,没必要把手里的对 12 先出来,对吧?


 

再比如 321 已经是「1、2、3」 这三个数字组合中最大的那个了,那就等于说没有更大的了,所以返回 123 这个最小的。


 

明白了吧?


 

要想解题,首先得明白题意,所以语文理解是非常重要的,其次是经验(😁)。


 

那凭借我们以往的经验,可能一下子会想到全排列。就拿 1、2、3、4 来举例吧,全排列如下:


nums=[1,2,3,4]

nums=[1,2,4,3]

nums=[1,3,2,4]

nums=[1,3,4,2]

nums=[1,4,2,3]

nums=[1,4,3,2]

nums=[2,1,3,4]

nums=[2,1,4,3]

nums=[2,3,1,4]

nums=[2,3,4,1]

nums=[2,4,1,3]

nums=[2,4,3,1]

nums=[3,1,2,4]

nums=[3,1,4,2]

nums=[3,2,1,4]

nums=[3,2,4,1]

nums=[3,4,1,2]

nums=[3,4,2,1]

nums=[4,1,2,3]

nums=[4,1,3,2]

nums=[4,2,1,3]

nums=[4,2,3,1]

nums=[4,3,1,2]

nums=[4,3,2,1]

观察nums = [1,3,4,2] -> nums = [1,4,2,3]这一步,因为4比3大,且4的位置在3之后,所以将4与3交换,必然能够使得nums变大,交换了之后,则变成了nums = [1,4,3,2]。


 

但显然这不是最小的比nums = [1,3,4,2]大的一个排列,我们还要把3和2的位置再翻转一下,才能得到nums = [1,4,2,3]这个 恰好 比nums = [1,3,4,2]大一点的排列。


 

那到底该怎么去找到这个 恰好 比nums大一点的排列呢?

第一步,我们可以从右向左查找第一个升序的相邻数字对 (i, i+1),满足 nums[i] < nums[i+1]。


 

这意味着从 i+1 到末尾的数字都是降序的。如果找不到这样的 i(即整个数组是降序的),这说明当前排列已经是最大的排列,我们只需将其翻转为最小排列即可。

这一步完成之后,并不能保证我们得到的排列就是 恰好 比nums大一点的排列。

这一步完成之后,并不能保证我们得到的排列就是 恰好 比nums大一点的排列。


 

第二步,我们还要对nums[i+1] 到 nums[nums.length - 1]这个区间进行翻转,使得它变成升序排列,这样才能得到 恰好 比nums大一点的排列。


 

比如说上面提到的 [1,4,3,2],i+1(i=1)到末尾的部分是 32。这部分是降序的。为了得到下一个排列,我们需要这部分变成升序。我们需要将这部分翻转,变成 23。

所以到这里,这道题目就迎刃而解了。具体代码实现:

class Solution {public void nextPermutation(int[] nums) {// 步骤1:从右向左查找第一个升序的相邻数字对(i, i+1),满足nums[i] < nums[i+1]。int i = nums.length - 2;while (i >= 0 && nums[i] >= nums[i + 1]) {i--;}if (i >= 0) {// 步骤2:在nums[i+1:]中从右向左找到第一个大于nums[i]的数字nums[j]。int j = nums.length - 1;while (j >= 0 && nums[i] >= nums[j]) {j--;}// 步骤3:交换nums[i]和nums[j]。swap(nums, i, j);}// 步骤4:将nums[i+1:]翻转,使其变为升序。reverse(nums, i + 1);}// 用于交换数组中两个元素的位置private void swap(int[] nums, int i, int j) {int temp = nums[i];nums[i] = nums[j];nums[j] = temp;}// 用于将数组的一部分翻转,即将nums[start:]变为升序private void reverse(int[] nums, int start) {int end = nums.length - 1;while (start < end) {swap(nums, start, end);start++;end--;}}
}


为了更清晰地理解题解代码,我们将其拆分成几个关键部分,并逐一说明每部分的作用和逻辑。


 

1.寻找升序对 (i, i+1)
int i = nums.length - 2;
while (i >= 0 && nums[i] >= nums[i + 1]) {i--;
}

  1. 目的:从数组的右端开始向左扫描,寻找第一个升序的相邻数字对,即找到第一个nums[i] < nums[i + 1]的位置。这个位置i是需要进行调整的起点,因为nums[i]右边的序列是降序的,没有更大的排列空间。
  2. 逻辑:使用一个while循环从右向左遍历数组,直到找到满足升序条件的i。


 

2.在 nums[i+1:] 中找到第一个大于 nums[i] 的数字并交换


 

nums[i+1:] 是指从i+1到数组末尾的部分。

if (i >= 0) {int j = nums.length - 1;while (j >= 0 && nums[i] >= nums[j]) {j--;}swap(nums, i, j);
}

  1. 目的:如果找到了这样的i,则在其右侧找到第一个大于nums[i]的数字nums[j],然后交换nums[i]和nums[j]。这一步是为了在当前排列的基础上得到下一个稍大的数字。
  2. 逻辑:通过向左扫描数组的剩余部分来查找j,一旦找到就执行交换。


 

3.翻转 nums[i+1:] 使其升序

reverse(nums, i + 1);

  1. 目的:交换nums[i]和nums[j]后,i之后的序列仍然保持降序。为了获得下一个排列,需要将这个序列翻转成升序,这样从i+1到数组末尾就构成了最小的排列,确保整个数组是下一个更大的排列。
  2. 逻辑:从i+1开始到数组末尾,执行翻转操作,使其成为升序。


 

4.交换方法 swap

private void swap(int[] nums, int i, int j) {int temp = nums[i];nums[i] = nums[j];nums[j] = temp;
}

交换数组中两个位置的元素。


 

5.翻转方法 reverse

private void reverse(int[] nums, int start) {int end = nums.length - 1;while (start < end) {swap(nums, start, end);start++;end--;}
}

将数组从指定位置start到数组末尾的部分翻转,使其成为升序。

测试:


/*** @ClAssName NextPermutation* @Description* 输入:nums=[1,2,3]* 输出:[1,3,2]* 输入:nums=[3,2,1]* 输出:[1,2,3]* 输入:nums=[1,1,5]* 输出:[1,5,1]* @Author 欧妮甲是神仙* @Date 2024/5/11 17:{MINUTE}*/
public class NextPermutation {public static void main(String[] args) {int[] nums = {1,3,4,2};nextPermutation(nums);System.out.println(Arrays.toString(nums));}static void  nextPermutation(int[] nums){//1、从右向左查找到第一个升序的相邻数字(i, i+ 1) ,满足 nums[i] < nums[i +1] .//那么一直找的条件是 反过来的, num[i]  > num[i +1]int i = nums.length -2;   ///表示倒数第二个数 ,如果为倒数第一个数, nums[i] >= nums[i+1] ,这里会数组越界while (i >=0 && nums[i] >= nums[i+1]){i--;}//找到定位的i之后if (i >=0){//2、在num[i+1]中从右向左找到第一个大于num[i]的数字num[j].。int j = nums.length-1;while (j >=0 && nums[i] >=nums[j]){j--;}//3、交换num[i]和num[j]swap(nums ,i , j);}//4、将num[i+1]翻转,使其变为升序revers(nums , i +1);}private static void  swap(int[] nums, int i, int j){int temp = 0;temp = nums[i];nums[i] = nums[j];nums[j] = temp;}//用于将数组的一部分翻转,即将nums[start] 变为升序private static void  revers(int[] nums, int start){int end = nums.length-1;while (start < end){swap(nums , start ,end);start++;end--;}}}


 

效率嘛,自然是快得飞起。

总结


 

这道题目的难点其实在于字典序的理解,一旦理解之后,整个题目的解法就不算是特别难了。


题目链接地址:

. - 力扣(LeetCode)

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

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

相关文章

移除链表元素(C语言)———链表经典算法题

题目描述&#xff1a;203. 移除链表元素 - 力扣&#xff08;LeetCode&#xff09; 答案展示: 答1&#xff08;遍历删除&#xff09;&#xff1a; /*** Definition for singly-linked list.* struct ListNode {* int val;* struct ListNode *next;* };*/ struct ListNo…

c++多态机制

多态 在 C 中&#xff0c;多态&#xff08;Polymorphism&#xff09;是一种面向对象编程的重要概念&#xff0c;它允许不同类的对象对同一消息做出不同的响应。具体来说&#xff0c;多态性允许基类的指针或引用在运行时指向派生类的对象&#xff0c;并且根据对象的实际类型来调…

第七届精武杯部分wp

第一部分&#xff1a;计算机和手机取证 1.请综合分析计算机和手机检材&#xff0c;计算机最近一次登录的账户名是 答案&#xff1a;admin 创建虚拟机时直接给出了用户名 2. 请综合分析计算机和手机检材&#xff0c;计算机最近一次插入的USB存储设备串号是 答案&#xff1a…

抖音快速涨粉秘籍解密!从巨量千川投流真实粉丝,快速增粉1000~10万!

随着抖音的风靡&#xff0c;对于众多用户来说&#xff0c;快速涨粉已经成为了追求的目标。在这篇文章中&#xff0c;我们将揭秘全网都在搜索的抖音快速涨1000粉的方法&#xff0c;帮助你打造一个高人气的抖音账号&#xff01;从巨量千川投流到官方真实流量&#xff0c;再到真实…

外卖系统微信小程序支付

微信小程序支付时序图 其中第9.步骤就是微信小程序前端调用wx.requestPayment

QT7_视频知识点笔记_3_自定义控件,事件处理器⭐,定时器,QPainter,绘图设备,不规则窗口

第三天&#xff1a; 自定义控件&#xff0c;事件处理器⭐&#xff0c;定时器&#xff0c;QPainter,绘图设备&#xff0c;不规则窗口实现 1.自定义控件&#xff1a; 创建新的QT控件类&#xff0c;然后再需要使用的地方--》提升为 来使用如何使用基础控件的信号和槽函数&…

1.前端环境搭建

1.安装nodejs 因为我们开发Vue项目需要使用npm命令来创建和启动&#xff0c;安装node.js是为了获得这个命令&#xff0c;目前和使用node.js无关 下载地址&#xff1a;http://nodejs.cn/download/ 下载完之后安装&#xff0c;通过cmd查看是否安装成功 node --version2.创建项目…

探讨 vs2019 c++ 里函数指针与函数类型在使用上的语法区别

&#xff08;1&#xff09;咱们可以用 decltype &#xff08;&#xff09; 来判断函数的类型。但以这个类型定义有用的可指向已存在函数的变量&#xff0c;却行不通。测试如下&#xff1a; 如果把上面的注释去掉会报错&#xff1a; 所以函数类型只有语法意义。但在使用上没有函…

【C语言】/*操作符(下)*/

目录 一、操作符的分类 二、二进制和进制转换 2.1 进制 2.2 进制之间的转换 三、原码、反码、补码 四、单目操作符 五、逗号表达式 六、下标引用操作符[] 七、函数调用操作符() 八、结构体成员访问操作符 8.1 直接访问操作符(.) 8.2 间接访问操作符(->) 九、操作符…

修改el-checkbox样式

一定要在最外层&#xff1b; //未选中框/deep/ .el-checkbox__inner{border-color: #0862a3;}//选中框/deep/ .el-checkbox__input.is-checked .el-checkbox__inner{background-color: #0862a3;border-color: #0862a3;}//未选中框时右侧文字/deep/ .el-checkbox__label{}//选中…

git 推送github 选https遇到登录 openSSH问题

使用https需要使用github令牌token作为密码&#xff0c; 使用SSH不需要登录。 还有一个问题&#xff1a; 创建github仓库后没有quick setup页面解决办法 千万不要点击任何多的操作&#xff01;&#xff01;&#xff01;输入仓库名&#xff0c;直接create&#xff01;&#x…

return语句

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 return语句 一、return语句后面跟表达式二、return无返回三、return返回的值和函数返回类型不一致四、return语句执行后,后方仍然存在代码五、存在分支语句&#xff0c;需考虑…

去哪里找高清视频素材?推荐几个短视频素材免费网站

在数字时代&#xff0c;视频内容的质量直接影响观众的吸引力和留存率。尤其是高清、4K视频素材和可商用素材&#xff0c;它们在提升视觉质量和叙事深度方面起到了至关重要的作用。以下是一些国内外的顶级视频素材网站&#xff0c;它们提供的资源将为您的创作提供极大的支持和灵…

LeetCode/NowCoder-链表经典算法OJ练习1

目录 说在前面 题目一&#xff1a;移除链表元素 题目二&#xff1a;反转链表 题目三&#xff1a;合并两个有序链表 题目四&#xff1a;链表的中间节点 SUMUP结尾 说在前面 dear朋友们大家好&#xff01;&#x1f496;&#x1f496;&#x1f496;数据结构的学习离不开刷题…

机器人系统仿真

0、何为仿真 通过计算机对实体机器人系统进行模拟的技术。 1、为何仿真 低成本&#xff1a; 机器人实体一般价格昂贵&#xff0c;为降低机器人学习、调试的成本&#xff1b;高效&#xff1a; 搭建的环境更为多样且灵活&#xff0c;可以提高测试效率以及测试覆盖率&#xff1b…

三大消息传递机制区别与联系

目录 总结放开头 1、定义区别&#xff1a; EventBus Broadcast Receiver Notification 2、使用区别: EventBus Broadcast Receiver Notification 3、补充通知渠道&#xff1a; 通知渠道重要程度 总结放开头 BroadCast Receiver:属于安卓全局监听机制&#xff0c;接收…

【算法】最短路问题 bfs 到 dijkstra

1976、到达目的地的方案数 你在一个城市里&#xff0c;城市由 n 个路口组成&#xff0c;路口编号为 0 到 n - 1 &#xff0c;某些路口之间有 双向 道路。输入保证你可以从任意路口出发到达其他任意路口&#xff0c;且任意两个路口之间最多有一条路。 给你一个整数 n 和二维整…

五一 大项目--docker-compose编排lnmp完成wordpress

Docker 中的 Nginx 服务为什么要启用 HTTPS 一安装容器 1 安装docker-20.10.17 2 安装所需的依赖 sudo yum install -y yum-utils device-mapper-persistent-data lvm23 添加Docker官方仓库 sudo yum-config-manager --add-repo https://download.docker.com/linux/centos…

Linux-线程概念

1. 线程概念 线程&#xff1a;轻量级进程&#xff0c;在进程内部执行&#xff0c;是OS调度的基本单位&#xff1b;进程内部线程共用同一个地址空间&#xff0c;同一个页表&#xff0c;以及内存中的代码和数据&#xff0c;这些资源对于线程来说都是共享的资源 进程&#xff1a;…

RabbitMQ (windows) 安装

大家好我是苏麟 , 今天安装一下 RabbitMQ . 官网 : RabbitMQ: One broker to queue them all | RabbitMQ 1.点击 Getting Started 2. 点击 Docs 3.点击 Install And Upgrade 4.点击 installation via Chocolatory 5. 直接下载安装包 RabbitMQ 下好了先放在一遍 RabbitMQ 需要 E…