leetcode-60-Permutation Sequence

问题

题目:[leetcode-60]

思路

当然,用之前枚举的方法可以。不过,不出所料,TLE。毕竟当 k = n时, T(n) = n! = O(n!)

只能用另外的思路:这个问题牵扯到另外一个具体的问题。[全排列的编码与解码]可以参考这个链接,讲的比较清楚。我下文的内容也完全参照这个链接,只不过我再自己梳理一遍,两个例题即可。
本质还是利用了阶乘的公式:Fact( n ) = n * Fact( n - 1 )(n>=2时)

  • 全排列编码

如 {1,2,3} 按从小到大排列一共6个:123 132 213 231 312 321。想知道321是{1,2,3}中第几个大的数。
这样考虑:第一位是3,小于3的数有1、2 。所以有2*2!个。第二位数2,小于2的数只有一个就是1 ,所以有1*1!=1 。第三位时1,小于1的数没有,是0个。所以小于32的{1,2,3}排列数有2*2!+1*1! + 0*0!=5个。所以321是第6个大的数。2*2!+1*1!是康托展开。(注意判断排列是第几个时要在康托展开的结果后+1)

再举个例子:1324是{1,2,3,4}排列数中第几个大的数:第一位是1小于1的数没有,是0个,0*3!,第二位是3小于3的数有1和2,但1已经在第一位了,所以只有一个数2,1*2! 。第三位是2小于2的数是1,但1在第一位,所以有0个数,0*1!。第四位的数是4,比4小的数有3个,但是之前都出现过了。所以是0个。所以比1324小的排列有0*3!+1*2!+0*1! + 0*0!=2个,1324是第三个大数。

又例如,排列3 5 7 4 1 2 9 6 8展开为98884,因为X=2*8!+3*7!+4*6!+2*5!+0*4!+0*3!+2*2!+0*1!+0*0!=98884.

  • 全排列解码

如何找出第16个(按字典序的){1,2,3,4,5}的全排列?
    1. 首先用16-1,得到15(序列为16,证明前面有15个数)
    2. 用15去除4!,得到0余15
    3. 用15去除3!,得到2余3
    4. 用3去除2!,得到1余1
    5. 用1去除1!,得到1余0
  有0个数比它小的数是1,所以第一位是1,有2个数比它小的数是3,但1已经在之前出现过了所以是4,有1个数比它小的数是2,但1已经在之前出现过了所以是3,有1个数比它小的数是2,但1,3,4都出现过了所以是5,最后一个数只能是2,所以排列为1 4 3 5 2。

代码

代码需要注意两点:

  • 判断前面有几个数字,不能重复。
  • 当前数字是多少,不能重复。
class Solution {
public:
    string getPermutation(int n, int k) {
        if(1==n)
            return "1";

        // 构造阶乘数组
        const int maxn = n;
        int fact[maxn];
        fact[maxn - 1] = 1;
        for( int i = maxn - 2, base = 2; i >= 0; --i, ++base )
        {
            fact[i] = base * fact[i+1];
        }

        std::string ans(n, ' ');
        const int sz = n + 1;
        bool visit[sz] = {false};

        --k; // K之前的数字个数
        for(int i = 0; i < maxn - 1; ++i)
        {
            int pre_num = k / fact[i+1];
            k %= fact[i+1];

            int val = 1;
            for( val = 1; pre_num&&val <= n; ++val )// 判断前面数字
            {
                if( !visit[val] ) --pre_num;
            }
            while(visit[val]) ++val;// 当前数字是多少

            ans[i] = '0' + val;
            visit[val] = true;
        }
        for( int i = 1; i <= n; ++i )
        {
            if( !visit[i] )
            {
                ans[maxn - 1] = '0' + i;
                break;
            }
        }
        return ans;
    }
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值