码迷,mamicode.com
首页 > 其他好文 > 详细

打印全排列思路

时间:2017-09-09 00:49:52      阅读:244      评论:0      收藏:0      [点我收藏+]

标签:递归   统一   需要   不能   code   put   bsp   问题:   png   

从n个不同的元素中取m个元素(m<=n),按照一定的顺序排列起来,

叫做从n个不同元素取出m个元素的一个排列。

当m=n时,所有的排列情况叫做全排列,比如3的全排列为:

1 2 3
1 3 2

2 1 3
2 3 1

3 1 2
3 2 1


我们先从简单的开始,要求写出代码打印上面的排列情况即可,顺序可以不一致。

分析过程:

首先,我们如何把三位的数字打印出来呢,有两种方式:

printf("%d\n" ,num);  //num=123

第二种:

printf("%d%d%d\n",a,b,c); //a=1 b=2 c=3

我认为采用第二种比较好,原因在于第一种需要对位数的考虑,

而我们问题需要对数字位置不断的进行交换,因此第二种也方便于交换。

 

现在知道了如何打印,那么如何交换数字的位置呢?很简单:

void swap(int *a,int *b){
  int tmp = *a;
  *a = *b;
  *b = tmp;
}

 

于是将上面的交换打印结合起来就可以,大概写个模型了

void output(int a,int b,int c){
   printf("%d%d%d\n",a,b,c);
   swap(&a,&b);
output(a,b,c); //递归 }

 

如果此时执行 output(1,2,3),那么程序将会死循环打印出下面的片段

123
213
..

 

因此我们有两个问题要解决:

1.怎样交换使得所有全排列都能被遍历到?

2.怎样使得递归能够终止?

 

从上面的死循环输出可以观察到,

交换仅仅发生在 第一个数字第二个数字 之间。

如果我们尝试继续对 第二个数字 第三个数字 进行交换。

然后递归重复上面这个过程,

1 2 3   //初始值,第一个周期开始

2 1 3   //前两个数字做交换
2 3 1   //后两个数字做交换

...  递归  ...

3 2 1   //前两个数字做交换
3 1 2   //后两个数字做交换

...  递归  ...

1 3 2   //前两个数字做交换
1 2 3   //后两个数字做交换,进入下一个周期
...  递归  ...

... 无限循环 ...

你会发现所有的排列就可以被打印出来了,但是个死循环的打印。

 

如何终止?

只需要找到一个周期的开始特征即可,比如在上面的打印中

初始值是 1 2 3

而所有全排列情况打印完后,下个周期的开始也是 1 2 3

那么可以来个判断a b c 变量是否同时和一开始一样,如果是就退出函数返回。

 

解决两个问题后,修改一下output函数:

void output(int a,int b,int c){
   
   printf("%d%d%d\n",a,b,c); //打印初始值

   swap(&a,&b);    //交换前两个数字
   printf("%d%d%d\n",a,b,c);

   swap(&b,&c);    //交换后两个数字

   //一个周期结束的情况发生在这个位置
   if(a==1 && b==2)return;

   output(a,b,c);   //递归

}

 

然后在main函数里面 执行 output(1,2,3) 输出结果如下:

技术分享

当然为了好看,你可以把交换的顺序改下,先交换后面两个数字,再交换首尾两个数字

技术分享

 

现在,3的全排列打印出来了,现在我们尝试写个代码可以打印任意一个数的全排列

首先,假如打印4的全排列,如果继续使用上面的代码思路,那么将需要修改:

output(1,2,3)   =>  output(1,2,3,4);

printf("%d%d%d\n")  =>  printf("%d%d%d%d\n");  

 

很多地方需要像上面一样添加一个变量,这样的话打印100的全排列,岂不是要写100个变量 或者 参数??

这样很显然不科学,因此我们换一个问题:输入一个数字,然后打印这个数字的全排列。也就是:

上面原版是:输入三个数字,打印这三个数字全排列,output(1,2,3);

而现在改成:输入一个数字,打印这个数字的全排列,output(3),需要输出结果与上面相同。

 

现在参数变成了一个,要通过这个参数能将1 2 3 成员表示出来

要能交换这三个数字的其中两个,且不能使用变量多个变量,如何实现呢?

首先我们需要把 1 2 3 这三个数字通过 3 这个数字得到,并将其保存到数组里,很简单:

int i,arr[1024];
for(i=1;i<=3;i++)
   arr[i] = i;   //为了方便操作我们从1开始存储

 

然后遍历数组,对这个数组遍历输出一次,然后交换一次

int i,j;
for(i=1;i<3;i++){
 
   //输出当前的数字组合
   for(j=1;j<=3;j++)
      printf("%d",arr[j]);
   printf("\n");

   //交换数组两个数字
   swap(&arr[i],&arr[i+1]);

}

 

我们尝试执行 output(3)  ,结果输出如下

技术分享

 

我们稍微做下修改,

1、当 i 遍历到数组的最后一个元素时,与第一个元素交换

2、将 3 统一用参数n来表示

于是output函数就可以写成

void output(int n){
  int i,j,arr[1024];
  for(i=1;i<=n;i++)    //保存n的全排列成员
     arr[i] = i;

  for(i=1;i<=n;i++){   //遍历数组

//输出当前数组组合
for(j=1;j<=n;j++) printf("%d",arr[j]); printf("\n");

//交换数组的成员位置 if(i==n) swap(&arr[i],&ar
r[1]); //当遍历到最后一个时,与第一个成员做交换
else swap(
&arr[i],&arr[i+1]); //否则与下一个元素做交换 } }

 

那么假如n=3,将数组遍历一次的过程如下:

1 2 3  //初始状态 , i=1
2 1 3  //交换,i=2
2 3 1  //交换,i=3
1 3 2  //交换,i=4,退出循环

 

很显然遍历一次是不足以将所有的排列情况输出出来。

为了使得数组能够继续从下标1为开始遍历,思路就是使用递归

也就是第一次遍历完后,将遍历完后的数组 递归传给函数本身继续遍历

而要保持数组的状态,意味着数组要以参数的形式传递,其次数组的初始化不能再递归里初始。

所以初始化可以放在main函数里,当然既然用到了递归,就需要防止无休止的递归,也就是要找到退出状态

不妨我们再写代码之前分析一下:

传入数组
arr = [0, 1, 2, 3]  //0下标不使用,下面忽略

//开始遍历
[ 1, 2, 3]    // i=1,输出然后交换
[ 2, 1, 3]    // i=2,输出然后交换
[ 2, 3, 1]    // i=3,输出然后与第一个元素交换
[ 1, 3, 2]    // i=4,  退出循环

-----递归-----

传入数组
arr = [0, 1, 3, 2]  //保存上次的状态继续任务,0继续忽略

//开始遍历
[ 1, 3, 2]    // i=1 输出然后交换
[ 3, 1, 2]    // i=2 输出然后交换
[ 3, 2, 1]    // i=3 输出然后与第一个元素交换
[ 1, 2, 3]    // i=4 退出循环,至此我们不需要继续执行了

---因此需要在此写个递归结束判断条件----

 

递归的结束条件与上面思路一样,如果数组与最最开始的状态是一样的

那么表示一个周期已经完成:

int tag = 1;          //标志位
for(i=1;i<=n;i++){
   if(arr[i]!=i){     //如果存在不同继续递归
     tag = 0;
     break;
  }
}
if(tag)return //如果全相同退出递归

 

那么总的代码如下:

#include <stdio.h>
#include <stdlib.h>
#define MAX 1024

void swap(int *a,int *b){
  int tmp = *a;
  *a = *b;
  *b = tmp;
  return;
}
void output(int n,int arr[]){
  int i,j;
  for(i=1;i<=n;i++){

    //print
    for(j=1;j<=n;j++){
      printf("%d ",arr[j]);
    }
    printf("\n");

    //swap
    if(i==n)swap(&arr[i],&arr[1]);
    else swap(&arr[i],&arr[i+1]);

  }

  int tag = 1;
  for(i=1;i<=n;i++)
    if(arr[i]!=i){tag=0;break;}

  if(tag)return;

  output(n,arr);
}

int main(){
  int i;
  int n,arr[MAX];
  printf("enter a num:");
  scanf("%d",&n);
  for(i=1;i<=n;i++)
    arr[i]=i;

  output(n,arr);
  return 0;
}

执行结果

技术分享

 

打印全排列思路

标签:递归   统一   需要   不能   code   put   bsp   问题:   png   

原文地址:http://www.cnblogs.com/demonxian3/p/7496813.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!