新解全排列的递归实现

 我们在手动计算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;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值