leetcode60-Permutation Sequence(求指定位置的排列)

问题描述:

The set [1,2,3,…,n] contains a total of n! unique permutations.

By listing and labeling all of the permutations in order,
We get the following sequence (ie, for n = 3):

"123"
"132"
"213"
"231"
"312"
"321"

Given n and k, return the kth permutation sequence.

Note: Given n will be between 1 and 9 inclusive.

给你整数n和k,求出序列[1…n]组成的第k个排列。

问题求解:

对于没有重复数字的全排列,利用康托展开公式:

X=an*(n-1)!+an-1*(n-2)!+...+ai*(i-1)!+...+a2*1!+a1*0!

ai为整数,并且0<=ai<i(1<=i<=n)

只要O(n)的时间复杂度。

例一:n=5的序列[1,2,3,4,5],求第k=16个排列。

16个也就是要前面有15个数,要调用15next_permutation函数。

根据康托公式,15/4! = 0...15 =》有0个数比他小的数是1,所以第1位是1

拿走刚才的余数15,用15/3! = 2...3 =》剩下的数里有两个数比他小的是
41已经没了),所以第二位是4

拿走余数3,用 3/2! = 1...1 =》剩下的数里有一个数比他小的是3,所以第
三位是3

拿走余数1,用 1/1! = 1...0 =》剩下的数里有一个数比他小的是 5(只剩25了),所以第四位是5

拿走余数0,用 0/0! = 0...0 =》只剩下一个是2,加入即可.
所以排列是 1,4,3,5,2

例二:已知是n = 5,求14352是它的第几个序列?(同一道题)

用刚才的那道题的反向思维:

第一位是1,有0个数小于1,即0* 4!
第二位是4,有2个数小于4,即2* 3!
第三位是3,有1个数小于3,即1* 2!
第四位是5,有1个数小于5,即1* 1!
第五位是2,不过不用算,因为肯定是0

所以14352是 n = 5 的第 0 + 12 + 2 + 1 + 0 = 15 + 1(求的是
第几个,所以要加1) = 16。跟刚才那道题一样。

以上参考:http://blog.csdn.net/modiziri/article/details/22389303?utm_source=tuicool&utm_medium=referral

代码:

class Solution {
public:
    string getPermutation(int n, int k) {
        vector<int> nums;
        for(int i=0;i<n;i++)
        {//(1)给nums赋值
            nums.push_back(i+1);
        }
        string res="";
        int pre=k-1;//(2)求第k个,则前面有k-1个
        for(int i=n-1;i>=0;i--)
        {//(3)利用康托展开公式
            int j = pre/fac(i);//3.1求得商j表示有j个数比排在当前位的数小
            pre = pre%fac(i);//3.2将余数作为分子继续循环
            res.push_back(nums[j]+'0');//3.3加入到结果字符串
            nums.erase(nums.begin()+j);//3.4将已选择的数从数组删除
        }
        return res;
    }
    int fac(int n)
    {//求n的阶乘
        if(n<=1) return 1;
        int res=1;
        for(int i=2;i<=n;i++)
        {
            res *= i;
        }
        return res;
    }
};

由于本题n的范围固定,所以可以先求好n!对应放在数组,需要时按下标取出,这样可以进一步节省时间。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值