全排列算法原理和实现

简介: 本文为原创,如需转载,请注明作者和出处,谢谢! 全排列是将一组数按一定顺序进行排列,如果这组数有n个,那么全排列数为n!个。现以{1, 2, 3, 4, 5}为 例说明如何编写全排列的递归算法。
本文为原创,如需转载,请注明作者和出处,谢谢!

全排列是将一组数按一定顺序进行排列,如果这组数有n个,那么全排列数为n!个。现以{1, 2, 3, 4, 5}为

例说明如何编写全排列的递归算法。
1、首先看最后两个数4, 5。 它们的全排列为4 5和5 4, 即以4开头的5的全排列和以5开头的4的全排列。
由于一个数的全排列就是其本身,从而得到以上结果。
2、再看后三个数3, 4, 5。它们的全排列为3 4 5、3 5 4、 4 3 5、 4 5 3、 5 3 4、 5 4 3 六组数。
即以3开头的和4,5的全排列的组合、以4开头的和3,5的全排列的组合和以5开头的和3,4的全排列的组合.
从而可以推断,设一组数p = {r1, r2, r3, ... ,rn}, 全排列为perm(p),pn = p - {rn}。
因此perm(p) = r1perm(p1), r2perm(p2), r3perm(p3), ... , rnperm(pn)。当n = 1时perm(p} = r1。
为了更容易理解,将整组数中的所有的数分别与第一个数交换,这样就总是在处理后n-1个数的全排列。
算法如下:

#include  < stdio.h >   

int  n  =   0 ;  

void  swap( int   * a,  int   * b) 
{     
    
int  m;     
    m 
=   * a;     
    
* =   * b;     
    
* =  m; 
}  
void  perm( int  list[],  int  k,  int  m) 
{     
    
int  i;     
    
if (k  >  m)     
    {          
        
for (i  =   0 ; i  <=  m; i ++ )             
            printf(
" %d  " , list[i]);         
        printf(
" /n " );         
        n
++ ;     
    }     
    
else      
    {         
        
for (i  =  k; i  <=  m; i ++ )         
        {             
            swap(
& list[k],  & list[i]);             
            perm(list, k 
+   1 , m);             
            swap(
& list[k],  & list[i]);         
        }     
    } 

int  main() 
{     
    
int  list[]  =  { 1 2 3 4 5 };     
    perm(list, 
0 4 );     
    printf(
" total:%d/n " , n);     
    
return   0

谁有更高效的递归和非递归算法,请回贴。


国内最棒的Google Android技术社区(eoeandroid),欢迎访问!

《银河系列原创教程》发布

《Java Web开发速学宝典》出版,欢迎定购

目录
相关文章
|
8月前
|
算法 C++ 索引
【算法】——全排列算法讲解
【算法】——全排列算法讲解
288 0
|
8月前
|
机器学习/深度学习 存储 算法
【算法训练-回溯算法 一】【排列问题】全排列、全排列II
【算法训练-回溯算法 一】【排列问题】全排列、全排列II
107 0
|
3月前
动态规划——斐波那契模型
本文介绍了动态规划的基本步骤及其在几个典型问题中的应用。首先概述了动态规划的四个关键步骤:状态表示、状态转移方程、初始化及填表顺序,并说明了如何确定返回值。接着通过具体实例讲解了第 N 个泰波那契数、三步问题、使用最小花费爬楼梯以及解码方法等问题的求解过程,详细阐述了每一步的具体实现方法与代码示例。最后还讨论了如何优化初始化过程以简化编码。
38 4
动态规划——斐波那契模型
|
8月前
|
存储 机器学习/深度学习 人工智能
【算法基础】基础算法(三)--(双指针算法、位运算、离散化、区间合并)
【算法基础】基础算法(三)--(双指针算法、位运算、离散化、区间合并)
|
8月前
|
算法 测试技术 C#
【状态压缩 动态规划 数论】1799. N 次操作后的最大分数和
【状态压缩 动态规划 数论】1799. N 次操作后的最大分数和
|
算法
Manachar算法(马拉车算法):快速求取最长回文子串
求取最长回文子串的长度的最佳方法为 Manachar算法 ,俗称马拉车算法。常见的方法就是中心扩散法,但时间复杂度较高。
104 0
Manachar算法(马拉车算法):快速求取最长回文子串
|
自然语言处理 算法
经典算法之——解决全排列问题以及详解
经典算法之——解决全排列问题以及详解
262 0
|
算法 Java
算法打卡Day25_leetcode _461. 汉明距离
算法打卡Day25_leetcode _461. 汉明距离
算法打卡Day25_leetcode _461. 汉明距离
|
算法 Java Linux
【基础算法】二分例题(我在哪?)
【基础算法】二分例题(我在哪?)
124 0
【基础算法】二分例题(我在哪?)
经典算法之二分法
经典算法之二分法
  经典算法之二分法