剑指offer:面试题38:字符串的排列

题目

输入一个字符串,打印出该字符串中字符的所有排列。例如输入字符串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 result = new ArrayList<>();
    public ArrayList 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()进行排序。

你可能感兴趣的:(剑指offer)