题目
输入一个字符串,打印出该字符串中字符的所有排列。例如输入字符串abc,则打印出由字符a、b、c所能排列出来的所有字符串abc、acb、bac、bca、cab和cba。(本文代码采用ArrayList接收返回的字符串,并要求不出现重复字符串)
思路
将字符串看成两部分,一部分是第一个字符,另一部分是后面的所有字符。
首先确定第一个字符,该字符可以是字符串中的任意一个;固定第一个字符后,求出后面所有字符的排列(相同步骤,采用递归)。
实现第一个字符的改变,只需要将第一个字符和后面所有字符进行交换即可(最早自己想的是从原始字符串拿出第i个字符,然后合并剩下的字符到后面,其实就是个交换的过程,自己开始时想得太复杂了)。要记得字符串输出后要将字符交换回来,变回原始的字符串。
测试用例
1.功能测试(有多个重复字母的字符串、所有字符相同的字符串、一个字符或者多个字符的普通字符串)
2.特殊测试(字符串为null、“”)
Java实现
import java.util.ArrayList;
import java.util.Collections;
// 面试题 38 字符串排序
public class Permutation {
private ArrayList<String> result = new ArrayList<>();
public ArrayList<String> Permutation(String str) {
if (str.length() == 0) {
return result;
}
StringBuilder newStr = new StringBuilder(str);
Permutation(newStr, 0);
Collections.sort(result); // ArrayList 的排序方法
return result;
}
public void Permutation(StringBuilder str, int start) {
if (start == str.length() - 1 && !result.contains(str.toString())) {
result.add(new String(str.toString()));
System.out.println(str.toString());
} else {
for (int i = start; i < str.length(); i++) {
char tmp = str.charAt(i);
str.setCharAt(i, str.charAt(start));
str.setCharAt(start, tmp);
Permutation(str, start + 1);
tmp = str.charAt(i);
str.setCharAt(i, str.charAt(start));
str.setCharAt(start, tmp);
}
}
}
public static void main(String[] args) {
Permutation permutation = new Permutation();
String str = "abc";
permutation.Permutation(str);
}
}
收获
1.这里利用递归实现字符串的全排列
2.题目要求不能有重复字符串,因此在添加到ArrayList之前,先判断是否contain。
3.最后要求输出是按字母序的,因此使用Collection.sort()进行排序。