31.下一个排列

31.下一个排列

参考讲解视频:这个排列有点难嗷!|31. 下一个排列|回溯|算法|数据结构|编程|程序员|前端开发|java面试|华为认证|C++_哔哩哔哩_bilibili

思路:

优化实现:

已遍历得到的元素我不用通过一个candidate去进行保存,这些元素还是保存在数组中,我后续要用到的时候,只需要从swap_index的右边去进行查找来得到比大于nums[swap_index]下的最小数字

手撕Code

python 复制代码
class Solution:
    def nextPermutation(self, nums: List[int]) -> None:
        """
        Do not return anything, modify nums in-place instead.
        """
        
        n = len(nums)
        
        i = n - 2

        ### 第一步
            ## 逆序遍历,找到单调递减 nums[i+1] > nums[i] 的部分,此时表明了nums[i]这个元素可以被右边更大的元素所替代
            ## 当nums[i+1] > nums[i] 时,此时循环就终止了

        while i >= 0 and nums[i] >= nums[i+1]:      
            i -= 1
        
        ### 第二步
            ## 第一步已经保证了 [i+1:]的元素都是单调递减的了,此时我们要从这些元素中找到一个比nums[i]大的最小元素,去替换掉nums[i]
            ## 第一步跳出循环的方式有两种,一种是找到了nums[i+1] > nums[i] 的部分,一种是 i = -1; 后者则表示整个数组都是单调递减的,此时已经是最大的一个数字了。
            ## 即最后一个排列
        if i >= 0:
            for j in range(i+1, len(nums)):
                if nums[j] > nums[i]:
                    swap_index = j
            
            nums[i], nums[swap_index] = nums[swap_index], nums[i]


        ### 第三步,反转之后的部分
        left, right = i+1, len(nums)-1      ## 后续元素从单调递减改为单调递增,这样元素才会更小
        while left < right:
            nums[left], nums[right] = nums[right], nums[left]
            left += 1
            right -= 1

Code

python 复制代码
class Solution:
    def nextPermutation(self, nums: List[int]) -> None:
        """
        Do not return anything, modify nums in-place instead.
        """
        
        n = len(nums)

        i = n - 2       

        ## 第一步,找到nums[i+1] > nums[i]时 i 的位置
        while i >= 0 and nums[i] >= nums[i+1]:
            i -= 1
        
        ## 第二步,找到大于nums[i]的最小元素
        if i >= 0:              ## 如果 i < 0, 那就表示整个数组都是单调递减的,当前这个数字是最大的
            j = n - 1
            while j >= i+1:
                if nums[j] > nums[i]:       ### 找到了大于nums[i]的最小元素
                    nums[i], nums[j] = nums[j], nums[i]
                    break
                
                j -= 1

       ### 反转实现1 
        # left, right = i+1, len(nums)-1
        # while left < right:
        #     nums[left], nums[right] = nums[right], nums[left]
        #     left += 1
        #     right -= 1
        
       ### 反转实现2
        # nums[i+1:] = nums[i+1:][::-1]
    
       ### 反转实现3
        nums[i+1:] = reversed(nums[i+1:])
相关推荐
diediedei7 分钟前
C++编译期正则表达式
开发语言·c++·算法
夏鹏今天学习了吗18 分钟前
【LeetCode热题100(97/100)】二叉搜索树中第 K 小的元素
算法·leetcode·职场和发展
炽烈小老头28 分钟前
【 每天学习一点算法 2026/01/26】缺失数字
学习·算法
小桃酥ღ36 分钟前
[力扣每日习题][1339]. 分裂二叉树的最大乘积 2026.01.07
算法·leetcode·职场和发展
0思必得01 小时前
[Web自动化] 处理爬虫异常
运维·爬虫·python·selenium·自动化·web自动化
喵手1 小时前
Python爬虫零基础入门【第九章:实战项目教学·第17节】内容指纹去重:URL 变体/重复正文的识别!
爬虫·python·爬虫实战·python爬虫工程化实战·零基础python爬虫教学·内容指纹去重·url变体
谦宸、墨白2 小时前
从零开始学C++:二叉树进阶
开发语言·数据结构·c++
喵手2 小时前
Python爬虫零基础入门【第五章:数据保存与入库·第1节】先学最通用:CSV/JSONL 保存(可复现、可分享)!
爬虫·python·python爬虫实战·python爬虫工程化实战·python爬虫零基础入门·数据保存与入库·csv/jsonl
子夜江寒2 小时前
OpenCV 学习:图像拼接与答题卡识别的实现
python·opencv·学习·计算机视觉
hrrrrb2 小时前
【算法设计与分析】贪心算法
算法·贪心算法·代理模式