详细解答看C++输出全排列递归算法详细解释
假设数组含有n个元素,则提取数组中的每一个元素做一次头元素,然后全排列除数组中除第一个元素之外的所有元素,这样就达到了对数组中所有元素进行全排列的得目的。
比如 1,2,3.的全排列就是分别以1,2,3开始的全排列。
以1开始的全排列也就是2,3.的全排列,(2,3)的全排列就是分别以2和3开始的全排列。
设全排列R(n1,n2,n3.....nn),可以化简为分别以n1,n2,n3……开始的全排列。
即 n1R1(n2,n3.....nn),n2R2(n1,n3.....nn),n3R3(n1,n2,.....nn)……nnR(n1,n2,n3.....)
其中,R1(n2,n3.....nn)又可以按照R的方式继续进行……以此类推可以得到全排列。
#include <iostream>
#include<string.h>
using namespace std;
void swap(char *a, char *b){//交换两个数
char temp;
temp = *a;
*a = *b;
*b = temp;
}
int n = 0;
void perm(char list[], int k, int m){
if(k>m){//可以输出了
for(int i=0; i <= m; i++)
cout << " " << list[i];
cout << endl;
n++;
}else{
for(int i=k; i<=m; i++){
swap(&list[k], &list[i]);//首位字符与后面的每位进行交换
perm(list, k+1, m);//后面字符进行递归
swap(&list[k], &list[i]);//序列恢复原状
}
}
}
int main()
{
char list[100];
scanf("%s",list);
int len = strlen(list);
perm(list, 0, len-1);
cout << "total:" << n << endl;
return 0;
}
还有个更简单的方法。。。要相信,这种数学基础问题,总是会有前辈把函数都写好了。。。
C++ STL中提供了std::next_permutation与std::prev_permutation可以获取数字或者是字符的全排列,其中std::next_permutation提供升序、std::prev_permutation提供降序。
1.std::next_permutation函数原型
template <class BidirectionalIterator>
bool next_permutation (BidirectionalIterator first, BidirectionalIterator last );
template <class BidirectionalIterator, class Compare>
bool next_permutation (BidirectionalIterator first,BidirectionalIterator last, Compare comp);
说明:next_permutation,重新排列范围内的元素[第一,最后一个)返回按照字典序排列的下一个值较大的组合。
返回值:如果有一个更高的排列,它重新排列元素,并返回true;如果这是不可能的(因为它已经在最大可能的排列),它按升序排列重新元素,并返回false。
算法描述:
1、从尾部开始往前寻找两个相邻的元素
第1个元素i,第2个元素j(从前往后数的),且i<j
2、再从尾往前找第一个大于i的元素k。将i、k对调
3、[j,last)范围的元素置逆(颠倒排列)
#include<iostream>
#include<algorithm>
#define N 200
using namespace std;
int main(){
int input[N];
int n;
cin >> n;//输入字符长度
for(int i=0;i<n;i++)
cin >> input[i];
cout << endl;
sort(input, input+n);//排序
do{
for(int i=0;i<n;i++)
cout << input[i] << " ";
cout << endl;
}while(next_permutation(input, input+n));
return 0;
}
