我们在手动计算n个数的全排列的过程一般就是以第i(0<i<=n)个数为根,剩下的数构成根的孩子,孩子的定义和根的一样,如此递归下去,就会形成一棵树。最后就会形成一个森林,该森林的所有叶子个数就是全排列的个数,根到每个叶子的所有节点构成的序列就是一个排列。如1,2,3的全排列如下图所示:
在上图每幅图中,求以树根为开头的全排列,就是求其余的数的全排列,比如:要求1开头的全排列,求出2和3的全排列就行。因此,将树根依次和自己的孩子交换,再求其余的数的全排列,就可以求得最终结果。比如:将上图中第一棵树的根和其孩子交换,就可得到第二棵树和第三棵树。对于子树,也是同样的方法进行。如果树只有树根(即叶子节点),则表示一个新的全排列生成。据此,可得出以下递归算法。
#include <iostream>
#include <algorithm>
using namespace std;
//A中存放n个数,从根为n-k的数开始求全排列,k表示树的深度
void perm(int A[], int k, int n)
{
if(1 == k) //深度为1,则到了叶子节点,形成了一个新的全排列
{
for(int i=0; i<n; i++)
cout << A[i];
cout << endl;
}
else
{
for(int i=n-k; i<n; i++) //树根依次和自己的孩子交换
{
swap(A[n-k], A[i]); //交换
perm(A, k-1, n); //求新树的全排列
swap(A[n-k], A[i]); //再交换回来
}
}
}
int main()
{
cout << "1,2,3的全排列:" << endl;
int A[] = {1, 2, 3};
perm(A, sizeof(A)/sizeof(int), sizeof(A)/sizeof(int));
return 0;
}