题目 3030:
时间限制: 2s 内存限制: 192MB
题目描述
给定一个由不同的小写字母组成的字符串,输出这个字符串的所有全排列。
我们假设对于小写字母有‘a’ <‘b’ < ... <‘y’<‘z’,而且给定的字符串中的字母已经按照从小到大的顺序排列。
输入格式
只有一行,是一个由不同的小写字母组成的字符串,已知字符串的长度在1到6之间。
输出格式
输出这个字符串的所有排列方式,每行一个排列。要求字母序比较小的排列在前面。字母序如下定义:
已知S=s1s2...sk,T=t1t2...tk,则S<T等价于,存在p(1<=p<=k),使得s1=t1,s2=t2,...,sp−1=tp−1,sp<tp成立。
样例输入
abc
样例输出
abc acb bac bca cab cba
题目分析
从给出的测试样例就可以看出一组数的全排列,每个数都会有在开头的情况:
a b c => abc
acb
bac
bca
cab
cba
所以,我们的全排列可以看作这种思想:依次将每个字符放到开头,确定第一个字符后剩下的字符全排列,剩下的字符进入递归函数后再以确定第一个字符剩余字符进入递归函数......以此递归,出口当然是只剩下一个字符的情况。
因为这题涉及到一个字典序排序的情况,所以在将每个字符放到开头后不能是简单的交换,而是需要移动剩余的数。将字符放到开头是取出要放的字符,再将它之前的字符后移。一个字符开头的递归结束后需要回溯到初始状态,将开头字符取出,剩余字符前移,再将取出字符放回原位。
代码
#include<bits/stdc++.h>
using namespace std;
string str;
void swap_begin(string& str,int start,int end)//后移
{
char temp=str[end];
for(int i=end;i>start;i--)
{
str[i]=str[i-1];
}
str[start]=temp;
}
void swap_end(string& str,int start,int end)//前移
{
char temp=str[start];
for(int i=start;i<end;i++)
{
str[i]=str[i+1];
}
str[end]=temp;
}
//前后移动的方式是为了保障字典序升序排列。
void print_result(string& str,int start,int end)
{
for(int i=start;i<=end;i++)
{
cout<<str[i];
}
cout<<endl;
}
void prim(string str,int start,int end)//递归函数
{
if(start==end){
print_result(str,0,end);//出口,只剩一个字符的情况
}
else
{
for(int i=start;i<=end;i++)
{
swap_begin(str,start,i);//将每一个字符轮流换到开头
prim(str,start+1,end);//以特定字符开头的下一个字符开始递归
swap_end(str,start,i);//恢复原位
}
}
}
int main()
{
cin>>str;
int len=str.size();
prim(str,0,len-1);
return 0;
}