求字符数组的全排列算法

6 篇文章 0 订阅
5 篇文章 0 订阅

题目:输入一个字符串,打印出该字符串中字符的所有排列。
例如输入字符串abc,则输出由字符a、b、c 所能排列出来的所有字符串abc、acb、bac、bca、cab 和cba。

算法思想
1.把第1个字符换到最前面来(本来就在最前面),准备打印axx,再对后两个字符b和c做全排列。
2.把第2个字符换到最前面来,准备打印bxx,再对后两个字符a和c做全排列。
3.把第3个字符换到最前面来,准备打印cxx,再对后两个字符a和b做全排列。
可见这是一个递归的过程,把对整个序列做全排列的问题归结为对它的子序列做全排列的问题,注意我没有描述Base Case怎么处理,你需要自己想。你的程序要具有通用性,如果改变了N和数组a的定义(比如改成4个数的数组),其它代码不需要修改就可以做4个数的全排列(共24种排列)。

解题过程
1.当N = 1的时候,则直接打印数列即可。
2.当N = 2的时候,设数组为 [a, b]
打印a[0], a[1] (即a,b)
交换a[0],a[1]里面的内容
打印a[0],a[1] (此时已变成了[b, a] )
3.当N = 3的时候,数组为 [a, b, c]
3.1把a放在 a[0] 的位置(原本也是如此,a[0] = a[0]),打印b,c的全排列(即a[1], a[2]的全排列)
3.2把b放在a[0]的位置(这时候需要交换原数组的a[0]和a[1]),然后打印a, c的全排列,打印完后再换回原来的位置,即a还是恢复到a[0],b还恢复到a[1]的位置
3.3把c放在a[0]的位置(这时候需要交换的是原数组的a[0]和a[2]),然后打印a, b的全排列,打印完后再换回原来的位置,即a还是恢复到a[0],b还恢复到a[1]的位置
至此,全排列完成
当 N = 4,5,6,……的时候,以此类推。

我们可以这么做:以abc为例
固定a,求后面bc的排列:abc,acb,求好后,a和b交换,得到bac
固定b,求后面ac的排列:bac,bca,求好后,c放到第一位置,得到cba
固定c,求后面ba的排列:cba,cab。
这里写图片描述

#include <stdio.h>
void print(char * pStr, char * pBegin) {
        if (!pStr || !pBegin)
                return;
        if (*pBegin == '\0')
                printf("%s\n", pStr);
        else {
                for (char * pCh = pBegin; *pCh != '\0'; ++pCh) {
                        char tmp = *pCh;
                        *pCh = *pBegin;
                        *pBegin = tmp;
                        print(pStr, pBegin+1);
                        tmp = *pCh;
                        *pCh = *pBegin;
                        *pBegin = tmp;
                }
        }
}
int main(void) {
        char ch[] = {
                'a', 'b', 'c', '\0'
        };
        print(ch, ch);
        return 0;
}

这里写图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值