题目:给出n个互不相同的字符, 并给定它们的相对大小顺序,这样n个字符的所有排列也会有一个顺序. 现在任给一个排列,求出在它后面的第i个排列.
这是一个典型的康拓展开应用,首先我们先阐述一下什么是康拓展开。
(1)康拓展开
所谓康拓展开是指把一个整数X展开成如下形式:
X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[2]*1!+a[1]*0!。(其中,a为整数,并且0<=a[i]<i(1<=i<=n))
(2)应用实例
{1,2,3,4,...,n}表示1,2,3,...,n的排列如 {1,2,3} 按从小到大排列一共6个:123 132 213 231 312 321。他们间的对应关系可由康托展开来找到。
1 #include <iostream>
2 using namespace std;
3
4 void CantorReverse(int index,int *p,int n); //康托展开逆用,判断给定的位置中的排列
5 long int fac[]={1,1,2,6,24,120,720,5040,40320,362880}; //表示阶乘运算的结果
6 //long int fac[]={0!,1!,2!,3!,4!,5!,6!,7!,8!,9!};
7
8 int main(int argc,char *argv)
9 {
10 int len=5;
11 int *s=(int *)malloc(len*sizeof(int));
12 CantorReverse(96,s,len); //有数字{12345}组成的所有排列中,求出第96个排列的顺序
13 for(int i=0;i<len;i++)
14 cout<<s[i];
15 cout<<endl;
16 free(s);
17 return 0;
18 }
19 void CantorReverse(int index,int *p,int n)
20 {
21 index--; //勿丢
22 int i,j;
23 bool hash[10]={0};
24 for(i=0;i<n;i++)
25 {
26 int tmp=index/fac[n-1-i]; //tmp表示有tmp个数字比当前位置上的数字小
27 for(j=0;j<=tmp;j++)
28 if(hash[j]) tmp++;
29 p[i]=tmp+1;
30 hash[tmp]=1;
31 index%=fac[n-1-i];
32 }
33 return;
34 }