[LeetCode](面试题38)字符串的排列

题目

输入一个字符串,打印出该字符串中字符的所有排列。

你可以以任意顺序返回这个字符串数组,但里面不能有重复元素。

示例:

输入:s = "abc"
输出:["abc","acb","bac","bca","cab","cba"]

限制:

1 <= s 的长度 <= 8

解题思路

求字符串的排列,可以看成两步,第一步求所有可能出现在第一个位置的字符,即把第一个字符与后面所有的字符交换。第二部固定第一个字符,求后面所有字符的排列,也是分成两部分来解。

具体思路:先固定第 1 位字符、再固定第 2 位字符、… 、最后固定第 n 位字符。第一个字符起,每个字符分别与它后面的字符交换,如果后面的字符中有重复的字符,那么第一个字符仅与其中的一个进行交换,跳过其他重复的。在每次递归中,都设置一个hashset,记录交换过的字符,下次再遇到时,就跳过。

算法步骤:

  • 1)终止条件: 当 x = len© - 1 时,代表所有位已固定,则将当前组合 c 转化为字符串并加入 res,并返回;
  • 2)递推工作: 初始化一个 Set ,用于排除重复的字符;将第 x 位字符与 i∈[x,len©] 字符分别交换,并进入下层递归:
    • 2.1)若 c[i] 在 Set​ 中,代表其是重复字符,因此“剪枝”;
    • 2.2)将 c[i] 加入 Set​,以便之后遇到重复字符时剪枝;
    • 2.3)将字符 c[i] 和 c[x] 交换,即固定 c[i] 为当前位字符;
    • 2.4)开启下层递归:调用 dfs(x+1) ,即开始固定第x+1 个字符;
    • 2.5)撤销交换: 将字符 c[i] 和 c[x] 交换回来,以便下一个 c[i] 与 c[x] 交换。

代码

class Solution {
    public String[] permutation(String s) {
        char[] chs = s.toCharArray();
        List<String> res = new LinkedList<>();
        dfs(chs, 0, res);
        return res.toArray(new String[res.size()]);
    }

    private void dfs(char[] chs, int x, List<String> res){
        if(x==chs.length-1){
            res.add(String.valueOf(chs));
            return;
        }
        HashSet<Character> set = new HashSet<>();
        for(int i=x; i<chs.length; i++){
            // 重复跳过
            if(set.contains(chs[i])){
                continue;
            }
            set.add(chs[i]);
            // 交换
            swap(chs, x, i);
            dfs(chs, x+1, res);
            // 撤销交换
            swap(chs, x, i);
        }

    }

    private void swap(char[] chs, int a, int b){
        char temp = chs[a];
        chs[a] = chs[b];
        chs[b] = temp;
    }
}

你可能感兴趣的:(剑指offer刷题笔记)