问题
思路
当然,用之前枚举的方法可以。不过,不出所料,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;
}
};