全排列(分治)(Java语言 +全排列模板)

题目描述:

用交换的分治法实现前m(m<10)个自然数数的全排列。 提示:通过交换实现的全排列不是字典序的全排列。

输入格式:

输入一个数m,代表要用1-m个自然数的全排列。

输出格式:

输出前m个自然数的全排列。(每个数之间用一个空格隔开,每行结束后有一个空格。最后一行数后面有一个空行,即每输出一行后添加一个换行,不考虑是否是最后一行)

输入样例:

3

输出样例:

1 2 3 
1 3 2 
2 1 3 
2 3 1 
3 2 1 
3 1 2 

解题思路:

题目就是要求实现全排列(使用分治的思路),perm()函数为全排列的模板

分治思路:

分治法求解问题分为三个步骤:

  • 分解:将问题分为若干个子问题。
  • 解决:递归地求解每个子问题。
  • 合并:将每个子问题的解合并成为整个问题的解。

现在我们需要求具有n个元素的数组A的全排列。例如:大小为3的数组A=[a,b,c] ,其实应该是A=[‘a’,‘b’,‘c’]。下同),它的全排列为: [[a,b,c], [a,c,b], [b,a,c], [b,c,a], [c,a,b], [c,b,a]] 这是一个大小为 n*n 的二维数组。

使用分治算法求解全排列的过程如下 :

  • 分解:将数组分为子数组 A[1…k-1] 和一个元素 A[k]。 (1≤k≤n)
  • 解决:递归地求解每个子数组 A[1…k-1] 的全排列,直至子数组A[1…k-1]为空时结束递归。
  • 合并:将上一步的结果—A[1…k-1]的全排列(一个二维数组)与元素A[k]合并,得出A[1…k]的全排列。例如: [[]] 与 a 合并得到 [[a]] [[a]] 与 b 合并得到 [[a,b], [b,a]] [[a,b],[b,a]] 与 c 合并得到 [[a,b,c],[a,c,b],[c,a,b],[b,c,a],[c,a,b],[c,b,a]]

代码:

package text2;

import java.util.Scanner;

public class 全排列_分治 {
	static int n;
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Scanner scanner = new Scanner(System.in);
		n = scanner.nextInt();
		int arr[] = new int[n+1];
		for(int i=0;i<=n;i++)
			arr[i] = i;
		perm(arr, 1);
		
	}
	public static void perm(int arr[],int begin) {//确定了begin位置之前的 元素顺序
		if(begin==arr.length) {//生成了一种全排列
			for(int i=1;i<=n;i++) {
				System.out.print(arr[i]+" ");
			}
			System.out.println();
		}
		for(int i=begin;i<arr.length;i++) {
			int t = arr[i];
			arr[i] = arr[begin];
			arr[begin] = t;
			perm(arr, begin+1);
			//必须回溯,保证原来数组的内容
			t = arr[i];
			arr[i] = arr[begin];
			arr[begin] = t;
		}
	}

}

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值