STL next_permutation和prev_permutation

一、next_permutation的实现

**原理:**所谓的下一个排列实际就是按照字典顺序比该排列更大的一个排列。首先从后往前找一个递增序列(递增序列说明在该递增序列里这是按照字典排序的最大序列),并找到第一个非递增的索引 i。而为了得到一个字典序列更大的序列,就必须从递增序列中找一个稍微大于索引 i 对应的数值,并与它替换;并反转递增序列变为递减序列为更新后的最小序列,从而得到下一个排列。
next_permutation的实现经过了三个过程(为了方便说明,下面说的数组都是指STL能进行双向迭代的容器):
1.首先从i=last-1(数组的最后面)进行遍历,找到递增(从后往前看)数组的范围以及第一个非递增数组的位置;不是很好说,举个例子如下图所示。假设有个数组{0,2,4, 3,1},i从后往前遍历,找到第一个非递增的索引(遍历经过1,3,4递增序列),则递增序列的索引范围是[ ii ,last-1]。
在这里插入图片描述
2.然后再从后last-1往前遍历从递增序列(索引范围是[ ii, last-1 ])中找到一个比索引i对应(上图中的2)数大的数就就停止,并标记该索引值为j(上图中的3)。并交换索引i,j对应的数组的值。得到如图所示。
在这里插入图片描述
3.最后,反转递增序列{ 1, 2, 4}(索引为[ ii,last-1])为递减序列{4, 2, 1},如下图所示。
在这里插入图片描述
下面为STL next_permutation函数实现的一个版本:

template <class BidirectionalIterator>
bool next_permutation(BidirectionalIterator first, BidirectionalIterator last)
{
   
    if(first == last) return false; //空区间,不存在下一个排列
    BidirectionalIterator i = first;
    ++i;
    if(i == last) return false; //区间只有一个元素,不存在下一个排列
    i = last;
    --i;

    for(;;)
    {
   
        BidirectionalIterator ii = i; //ii为递增区间[ ii , last-1]
        --i;                          //i为第一个非递增区间的索引
        if(*i < *ii)
        {
   
            BidirectionalIterator j = last;
            while(!(*i < *--j));   //通过查找比*i稍微大的值*j,由于过滤了等于(*i == *j),索引该区间里可以存在重复的元素
            iter_swap(i, j);       //交换 *i 与 *j的值
            reverse(ii, last);     //反转递增区间
            return true;            //存在下一个排列
        }
        if(i == first)
        {
   
            reverse(first, last);   //如果i == first, 索引从last 到 first最大的递增序列,不存在下一个排列,反转区间
            return false;
        }
    }
}

在leetcode中不少的关于排列的题,都可以用到上面的思想进行实现,比如:
leetcode 31题

实现获取下一个排列的函数,算法需要将给定数字序列重新排列成字典序中下一个更大的排列。

如果不存在下一个更大的排列,则将数字重新排列成最小的排列(即升序排列)。

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

以下是一些例子,输入位于左侧列,其相应输出位于右侧列。
1,2,3 → 1,3,2
3,2,1 → 1,2,3
1,1,5 → 1,5,1

class Solution {
   
public:
    void nextPermutation(vector<int>& nums) {
   
        if(nums.empty() || nums.size() == 1)
            return;
        
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
C++的next_permutation一个STL函数,用于求一个序列的下一个字典序排列。它包含在头文件<algorithm>中。使用next_permutation函数可以方便地生成一个序列的所有可能排列。您可以通过传入一个数组和数组的大小来使用next_permutation函数。C++的STL库提供了next_permutation函数,它可以自动计算下一个排列,无需手动实现。要使用next_permutation函数,您需要包含<algorithm>头文件,并使用该函数进行迭代处理,直到没有下一个排列为止。手动实现next_permutation函数的方法是,首先找到一个逆序的位置,然后找到一个比该位置上元素大的元素,并将这两个元素交换位置,最后将逆序位置后面的元素颠倒顺序。这样就得到了下一个排列。通过不断重复这个过程,直到没有下一个排列为止。手动实现next_permutation函数的代码如下:<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [C++中 next_permutation 函数的使用方法、原理及手动实现](https://blog.csdn.net/m0_51913750/article/details/130540032)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *3* [详谈全排列next_permutation() 函数的用法(推荐)](https://download.csdn.net/download/weixin_38651365/13784181)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值