面试100题系列之5字符串的排列组合问题

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);
	}
}




 

你可能感兴趣的:(面试,递归)