数学建模社区-数学中国

标题: 输出一组数字的所有排列 [打印本页]

作者: kampoo    时间: 2005-12-29 14:29
标题: 输出一组数字的所有排列
输出一组数字的所有排列,最简单的算法是什么?
作者: fewrains    时间: 2006-1-1 13:15
<>递归,从VChelp的算法版主那里抄来的:</P>0 R0 }8 m  H% ?8 s$ ^3 N
<><BR>#define  N_permutation 5 // 排列元素数量</P>
5 M* \' N. R: `0 _& B$ ]" J) L<>unsigned char flag[N_permutation]; // 标记元素是否被使用<BR>unsigned char dat[N_permutation];  // 记录临时元素队列<BR>unsigned char p_dat=0;             // 对列指针</P>- {" v& H. I% V
<>void out_dat()   // 输出对列dat中的内容<BR>{<BR> for(int i=0;i&lt;N_permutation;i++)<BR>  cout&lt;&lt;dat&lt;&lt;' ';<BR> cout&lt;&lt;endl;<BR>}</P>; _$ _& E% ^& ^- }
<>void permutation()          // 再入函数,生成N_permutation个字母的所有排列,<BR>{                           // 最大递归次数为N_permutation<BR> unsigned char i;           // 被再入函数入栈保护的局部变量<BR> for(i=0;i&lt;N_permutation;i++) <BR> {<BR>  if (flag==0)<BR>  {<BR>   flag=1;                // 标记元素被使用<BR>   dat[p_dat]=0x41+i;        // 将字母添加到队列<BR>   p_dat++;                  </P>) R) U2 s% h  N( f# U, i3 Z6 I
<>   if (p_dat==N_permutation) // 对列满输出,<BR>    out_dat();<BR>   else permutation();       // 否则继续添加<BR>   p_dat--;<BR>   flag=0;<BR>  }<BR> } <BR>}<BR>void main()<BR>{<BR> permutation();<BR> getch();<BR>}</P>[em07][em07][em07]
作者: fewrains    时间: 2006-1-1 13:16
<>前头要加上:</P>5 [( ?( y7 g6 a
<>#include &lt;iostream&gt;<BR>#include &lt;conio.h&gt;<BR>using namespace std;<BR></P>
作者: fewrains    时间: 2006-1-3 00:44
<>非递归实现,效率不错</P>% v1 x8 c: t1 a
<>#include &lt;iostream&gt;<BR>#include &lt;conio.h&gt;<BR>#include &lt;windows.h&gt;<BR>using namespace std;</P>
+ e8 c' p! o3 i; o6 G" R+ b: ~<>void output(int *p);     //输出函数<BR>void permutation(int *p);//全排列函数</P>
) N$ k: B' ~' h' V; D8 x. h<>int a[]={1,2,3,4,5,6,7,8,9,10,11};//待排列数组<BR>int N=sizeof(a)/sizeof(int);      //数组长度<BR>int *p=a;                         //数组指针</P>
1 B3 E( d/ t8 H) ]1 q<>void main()<BR>{<BR> permutation(p);    //调用全排列函数<BR> output(p);         //输出排列结果<BR> getch();<BR>}</P>
/ ~- F5 y- g+ m7 R" n<>void permutation(int *p)<BR>{<BR> bool flag=false;   //标志排列是否结束<BR> int temp;          //临时变量,用来交换数组元素</P>- p' o( A, `9 [. {. o
<> while(1)<BR> {<BR>  for(int i=0;i&lt;N-1;i++)<BR>  {<BR>   flag=true;     //如已排列完,falg为真</P>8 ~/ l7 ?8 l+ c3 @. ]2 l
<>   if(p&lt;p[i+1]) //如果发现第一个倒序的数组元素,p[i+1]被称为倒序元素<BR>   { <BR>    flag=false; //说明未排列完</P>% \+ C9 |2 ]/ j) }& N1 I
<>    int j=0;<BR>    while(p[j++]&gt;=p[i+1]); //找到第一个大于或等于倒序元素的数组元素<BR>    j-=1;                  </P>
' a9 W. r. O+ [0 u$ f6 b<>    temp=p[i+1];           //交换这两个元素<BR>    p[i+1]=p[j];<BR>    p[j]=temp;</P>; A* h( b, a5 I2 s$ y3 M, I7 y
<>    j=0;<BR>    while(j&lt;(i+1)/2)       //倒置倒充元素原位置以前的所有元素<BR>    {<BR>     temp=p[j];<BR>     p[j]=p[i-j];<BR>     p[i-j]=temp;<BR>     <BR>     j++;<BR>    }    <BR>    break;                 //跳出for循环,<BR>   }<BR>  }<BR>  if(flag)                       //如果排列完,结束循环<BR>   break;<BR>  //output(p);                        //输出排列结果<BR> }<BR>}</P>
2 ^' D3 m; |0 m3 q1 N% z<>void output(int *p)<BR>{<BR> cout&lt;&lt;endl;<BR> for(int i=0;i&lt;N;i++)<BR>  cout&lt;&lt;p&lt;&lt;' ';</P>
: @6 D, I* X# O* c3 b/ c<>}<BR></P>
作者: dsg333    时间: 2006-2-19 05:16
还有吗?
作者: zhouming09    时间: 2006-3-6 17:01
太高深了 不东
作者: zansan    时间: 2006-5-7 19:49
<p>&nbsp;<font color="#000000" size="+0">递归容易明白,就是慢,这个算法就是好.找到"倒置位"后进行倒置.</font></p>
作者: jinfly4997    时间: 2006-8-11 20:22
matlab里面好像有这个函数,我用过,但现在实在是既不起来了,你可以寻求帮助,但如果英语不太好,建议买一本参考教材看看相关命令,或者上网查一些有关命令的资料下载就是了,我的大部分matlab知识就是寻求网上帮助获得的,你也可以试试!
作者: sjian2001    时间: 2006-10-21 22:16
看过一篇文章是这么说的:<br/>比如说求1,2,3的全排列,<br/>先是{  1  }<br/>然后用2,在元素里所有空当插入<br/>变成了{ 21  , 12}<br/>然后用3,在元素里所有空当插入<br/>{321,231,213,312,132,123}<br/>算法还是好实现的。<br/>不知道跟其他算法有什么效率差别。<br/>




欢迎光临 数学建模社区-数学中国 (http://www.madio.net/) Powered by Discuz! X2.5