全排列(递归方法)

题目 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;
 }

                                                  

  • 4
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值