C实现字符排列

简介: 版权声明:您好,转载请留下本人博客的地址,谢谢 https://blog.csdn.net/hongbochen1223/article/details/46437485 用已知字符串s中的字符,生成由其中n个字符组成的所有字符的排列。
版权声明:您好,转载请留下本人博客的地址,谢谢 https://blog.csdn.net/hongbochen1223/article/details/46437485

用已知字符串s中的字符,生成由其中n个字符组成的所有字符的排列。设n小于字符串s的字符个数,其中s中的字符在每个排列中最多出现一次。 例如,对于s[]=”abc”,n=2,则所有字符的排列有:ba,ca,ab,cb,ac,bc。

算法思想:
使用递归完成该实例。

  • 举个例子:
  • s = “abc”,n=2
  • 则第一个perm(n,s),即perm(2,”abc”);
  • 首先需要判断w中的字符个数是否满足,n=2>1,表示还没有满足
  • 首先,从s的第一个元素开始,s[1] = ‘a’;
  • 填充到w中,w[n-1]即,w[1] = ‘a’;
  • 紧接着,进行一些调整,使得s1 = bc,n=1
  • 进行同样的perm,即perm(1,”bc”);同样先进行判断
  • 选择b填充到w中,在进入perm(0,”c”),判断输出
  • 接着,回到上一层,perm(1,”bc”),再选择字符”c”进入w
  • 选择c进入w之后,进行一些调整,将s1调整成这样一个字符串s1=”cb”,
  • 这样进入perm(0,”b”),判断输出.
    *
  • 接着,进入第一层,由于ba,ca都已经输出完毕,第一个元素从b开始进行选择
  • 首先将b放入w中,即w[1] = ‘b’;接着进行一些调整,将s1调整为s1 = “bac”;
  • 进入perm(1,”ac”);
  • 以此方式进行递归,完成。

下面是代码实现部分:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define N 20
char w[N];

void perm(int n,char *s);

/**
 * 用已知字符串s中的字符,生成由其中n个字符组成的所有字符的排列。
 * 设n小于字符串s的字符个数,其中s中的字符在每个排列中最多出现一次。
 * 例如,对于s[]="abc",n=2,则所有字符的排列有:ba,ca,ab,cb,ac,bc。
 */
int main()
{
    char s[N];
    int n;

    printf("Please enter the char array:\n");
    scanf("%s",&s);
    printf("Please number:\n");
    scanf("%d",&n);

    perm(n,s);  //调用排列函数

    return 0;
}

/**
 * 举个例子:
 * s = "abc",n=2
 * 则第一个perm(n,s),即perm(2,"abc");
 * 首先需要判断w中的字符个数是否满足,n=2>1,表示还没有满足
 * 首先,从s的第一个元素开始,s[1] = 'a';
 * 填充到w中,w[n-1]即,w[1] = 'a';
 * 紧接着,进行一些调整,使得s1 = bc,n=1
 * 进行同样的perm,即perm(1,"bc");同样先进行判断
 * 选择b填充到w中,在进入perm(0,"c"),判断输出
 * 接着,回到上一层,perm(1,"bc"),再选择字符"c"进入w
 * 选择c进入w之后,进行一些调整,将s1调整成这样一个字符串s1="cb",
 * 这样进入perm(0,"b"),判断输出.
 *
 * 接着,进入第一层,由于ba,ca都已经输出完毕,第一个元素从b开始进行选择
 * 首先将b放入w中,即w[1] = 'b';接着进行一些调整,将s1调整为s1 = "bac";
 * 进入perm(1,"ac");
 * 以此方式进行递归,完成。
 */
void perm(int n,char *s){
    if(n < 1){
        //如果n小于1,表示w中的长度已经达到需要的长度
        printf("%s\n",w);
        return;
    }else{
        char s1[N];  //临时变量存储
        strcpy(s1,s);

        int i;

        for(i = 0;*(s1+i);i++){
            //从s1中选择一个字符放入w对应的位置中
            *(w+n-1) = *(s1+i);
            //将从s1中被选择的字符替换成第一个字符
            *(s1+i) = *s1;
            //将第一个字符换成被选择的字符
            *s1 = *(w+n-1);
            perm(n-1,s1+1);
        }

    }

}

下面是程序的运行结果:

这里写图片描述

总体来说,这个算法的思路还是比较简单的。

目录
相关文章
|
索引
【LeetCode】917. 仅仅反转字母、387. 字符串中的第一个唯一字符
目录 917. 仅仅反转字母 387. 字符串中的第一个唯一字符
54 0
|
7月前
|
算法
现有‘abcdefghijkl’12个字符,将其所有的排列按字典序进行排序,给出任意一组排列,说出这租排列在所有排列中是第几小的
现有‘abcdefghijkl’12个字符,将其所有的排列按字典序进行排序,给出任意一组排列,说出这租排列在所有排列中是第几小的
68 1
|
8月前
|
算法 C++
Acwing.51 数字排列(全排列)
Acwing.51 数字排列(全排列)
|
8月前
|
Java API C++
leetcode-151:翻转字符串里的单词
leetcode-151:翻转字符串里的单词
62 0
从排列字符串到排列序列:解析增减字符串匹配问题
题目要求根据给定的字符串 s,构造一个排列序列 perm,其中排列序列中的数字满足以下规则: 如果 perm[i] < perm[i + 1],则对应的字符为 'I'; 如果 perm[i] > perm[i + 1],则对应的字符为 'D'。 我们需要根据字符串 s 中的字符,构造满足上述规则的排列序列 perm。
70 0
剑指offer_字符串---字符串的排列
剑指offer_字符串---字符串的排列
61 0
LeetCode 567. 字符串的排列
LeetCode 567. 字符串的排列
96 0
|
算法 Java
翻转字符串里的单词 (LeetCode 151)
翻转字符串里的单词 (LeetCode 151)
161 0
|
存储 API 索引
leetcode【字符串—中等】151.翻转字符串里的单词
leetcode【字符串—中等】151.翻转字符串里的单词
leetcode【字符串—中等】151.翻转字符串里的单词