【Golang】LeetCode 31. 下一个排列

31. 下一个排列

题目描述

思路

这道题目已经不是第一次在我的频道出现了😂,但是时隔过久没有刷题,又忘记了思路,在此重新记录一次。

寻找下一个排列,也就是寻找下一个比当前稍大的组合。我们要做的就是从后往前首先找到一个"可提升的位置",也就是从后往前寻找一个"谷值"。如果从后往前我们不能找到一个位置满足nums[i] < nums[i + 1],这就说明当前数组已经是降序排列了,也就是"最后一个排列",直接将整个数组翻转,就可以得到下一个排列(比如[3, 2, 1],下一个排列就是[1, 2, 3])。

如果我们能够找到谷值,接下来我们需要做的就是"提升这个位置的大小"。一个很直观的想法就是:我们再次从后向前,找到第一个比nums[i]要大的数,满足nums[i] < nums[j]j这个位置的数就是我们要与i交换的数,交换之后可以立马提升i位置的大小。

不过到此还没有结束,以[4, 5, 6, 1, 3, 2]这个比较复杂的排列为例,显然nums[i] == 1; nums[j] == 2。如果单纯交换两个位置的数,得到[4, 5, 6, 2, 3,1],并不是下一个排列,显然[4, 5, 6, 2, 1, 3][4, 5, 6, 2, 3, 1]更小。所以我们在交换nums[i]nums[j]的数值之和,还需要显式地对区间[i + 1, n)进行一次翻转,以确保i位置之后的字典序最小。

至此,我们得到的序列就是输入序列的「下一个排列」。

Golang 题解

go 复制代码
func nextPermutation(nums []int)  {
    n := len(nums)
    i := n - 2
    // 首先找到一个"可提升的位置". 对于 [4, 5, 6, 1, 3, 2] 这个例子
    // 可提升的位置就是 1, 因为 1 < 3
    for i >= 0 {
        if nums[i] < nums[i + 1] {
            break
        }
        i --
    }
    // 如果找不到可提升的位置, 也就是最终 i < 0, 说明整个数组从前往后
    // 单调递减, 此时已经是最后一个排列, 直接反转整个数组得到下一个排列
    if i >= 0 {
        // 否则, 找到了一个可提升的位置, 此时从后到 i 寻找一个
        // 比 i 大的数值 j
        j := n - 1
        for nums[i] >= nums[j] {
            j --
        }
        // 交换 i, j 下标对应的这两个数
        // [4, 5, 6, 1, 3, 2] -> [4, 5, 6, 2, 3 ,1]
        // 但此时仍然不是下一个排列, 应该翻转 [i + 1, n) 这个区间
        // [4, 5, 6, 2, 1, 3] 才是 [4, 5, 6, 1, 3, 2] 的下一个排列
        nums[i], nums[j] = nums[j], nums[i]
    }
    for k, t := i + 1, n - 1; k < t; k, t = k + 1, t - 1 {
        nums[k], nums[t] = nums[t], nums[k]
    }
}
相关推荐
叫我:松哥19 分钟前
基于Python flask的中学可控智能命题系统设计与实现,整合遗传算法、DeepSeek 大模型及数据库技术构建一体化应用
数据库·人工智能·python·算法·机器学习·flask·遗传算法
CoderYanger25 分钟前
A.每日一题:3612. 用特殊操作处理字符串 I
java·程序人生·leetcode·面试·职场和发展·学习方法·改行学it
黎阳之光31 分钟前
黎阳之光透明大楼:实景孪生重构智慧建筑全新范式
人工智能·物联网·算法·安全·数字孪生
承渊政道32 分钟前
【MySQL数据库学习】(MySQL表的内外连接)
数据库·学习·mysql·leetcode·bash·数据库开发·数据库系统
旖-旎2 小时前
《LeetCode 130 被围绕的区域 FloodFill DFS 解法》
c++·算法·深度优先·力扣·floodfill
林森lsjs2 小时前
斐波那契数列的 N 种解法:从递归到动态规划的优化之路【算法思考】
算法·动态规划
apcipot_rain3 小时前
计科八股20260616(1)——堆存中位数、链表判环、黑白测试、敏捷开发与瀑布模型、配置管理、持续集成、池化
数据结构·算法·软件工程
JAVA面经实录9179 小时前
Java 数据结构与算法 (终极完整学习文档)
java·数据结构·算法
开源Z11 小时前
LeetCode 42 · 接雨水:从暴力到双指针的三步优化
算法·leetcode
旖-旎11 小时前
《LeetCode 695 岛屿的最大面积 FloodFill DFS 解法》
c++·算法·力扣·深度优先遍历·floodfill