1、字符串的组合
输入一个字符串,找出字符串中所有字符的所有组合形式。
比如说abcd,其组合形式有a,b,c,d,ab,ac,ad,bc,bd,cd,abc,abd,acd,bcd,abcd。这里认为adb和abd是一样的情况。组合的情况也就是遇到一个字符,分两种情况,取当前这个字符,或者不取当前这个字符。然后对于每一个字符都做这样的考虑,典型的递归思路。
void Range(char *a, int Index, char *ans, int num) { if(a[Index] == '\0')//字符数组遍历完之后,返回 { if(num > 0)//ans数组里面不为空,打印 Print(ans,num); return; } Range(a,Index + 1,ans,num);//不取Index位置上的字符 ans[num] = a[Index];//取Index位置上的字符 Range(a,Index + 1, ans, num + 1); }
2、字符串的排列
输入一个字符串,找出字符串中所有字符的所有排列形式。
比如说abc,其组合形式有abc,acb,bac,bca,cba,cab。这里以一个长一点的字符串abcdefg为例,假设ab已经排好了,现在排c,那这c就可以在cdefg这几个字符的任何一个位置,也就是后面的任意一个字符都可以出现在c的位置,进一步就是任意的交换c与后面字符的值都得到一种排列,这样相当于第三个位置(也就是c本来的位置)上的字符就确定了就确定了,然后再对得到的序列的下一个字符做同样的操作。典型的用递归求解。
void Rank(char *str, int b, int e) { if(!str || b > e || b < 0) return; int i; if(b == e - 1) { for(i = 0; i < e; ++i) printf("%c", str[i]); printf("\n"); return; } for(i = b; i < e; ++i) { swap(str[b], str[i]); Rank(str, b + 1, e); swap(str[b], str[i]); } }
3、题目扩展:其实上面的思路是有BUG的,为什么呢?试试字符串中有重复的字符,是不是得到了你想要的结果。解决的办法有一个,写一个函数移除重复的字符,或者如果不允许改变数据的话,就用另一个字符串来存储那些不重复的字符。
4、最后给出上面的一些辅助函数以及main函数的额调用:
#include<stdio.h> #include<string.h> char a[100]; char ans[100]; void Print(char *ans, int num) { for(int i = 0; i < num; ++i) putchar(ans[i]); printf("\n"); } inline void swap(char &a, char &b) { int t; t = a; a = b; b = t; } int main() { char str[20]; while(gets(str)) { Rank(str, 0, strlen(str)); //Range(a,0,ans,0); } }