Code Ganker: Permutations -- LeetCode

2014年3月19日星期三

Permutations -- LeetCode

原题链接: http://oj.leetcode.com/problems/permutations/
这道题跟N-QueensSudoku SolverCombination SumCombinations等一样,也是一个NP问题。方法还是原来那个套路,还是用一个循环递归处理子问题。区别是这里并不是一直往后推进的,前面的数有可能放到后面,所以我们需要维护一个used数组来表示该元素是否已经在当前结果中,因为每次我们取一个元素放入结果,然后递归剩下的元素,所以不会出现重复。时间复杂度还是NP问题的指数量级。代码如下:
public ArrayList<ArrayList<Integer>> permute(int[] num) {
    ArrayList<ArrayList<Integer>> res = new ArrayList<ArrayList<Integer>>();
    if(num==null || num.length==0)
        return res;
    helper(num, new boolean[num.length], new ArrayList<Integer>(), res);
    return res;
}
private void helper(int[] num, boolean[] used, ArrayList<Integer> item, ArrayList<ArrayList<Integer>> res)
{
    if(item.size() == num.length)
    {
        res.add(new ArrayList<Integer>(item));
        return;
    }
    for(int i=0;i<num.length;i++)
    {
        if(!used[i])
        {
            used[i] = true;
            item.add(num[i]);
            helper(num, used, item, res);
            item.remove(item.size()-1);
            used[i] = false;
        }
    }
}
注意在实现中有一个细节,就是在递归函数的前面,我们分别设置了used[i]的标记,表明改元素被使用,并且把元素加入到当前结果中,而在递归函数之后,我们把该元素从结果中移除,并把标记置为false,这个我们可以称为“保护现场”,递归函数必须保证在进入和离开函数的时候,变量的状态是一样的,这样才能保证正确性。当然也可以克隆一份结果放入递归中,但是那样会占用大量空间。所以个人认为这种做法是最高效的,而且这种方法在很多题目(几乎所有NP问题)中一直用到,不熟悉的朋友可以练习一下哈。
对于NP问题我们一直都是用递归方法,也是一个很成熟的套路,可以举一反三。不过有时候面试官会刻意让我们实现一下迭代的做饭,所以这里我们就介绍一下迭代的实现方法。迭代一般就是要理清每次加入一个新的元素,我们应该做什么,这里其实还是比较清晰的,假设有了当前前i个元素的所有permutation,当第i+1个元素加进来时,我们要做的就是将这个元素带入每一个之前的结果,并且放到每一个结果的各个位置中。因为之前的结果没有重复,所以带入新元素之后也不会有重复(当然这里假设数字集合本身是没有重复的元素的)。按照这个思路,实现的代码如下:
public ArrayList<ArrayList<Integer>> permute(int[] num) {
    ArrayList<ArrayList<Integer>> res = new ArrayList<ArrayList<Integer>>();
    if(num == null || num.length==0)
        return res;
    ArrayList<Integer> first = new ArrayList<Integer>();
    first.add(num[0]);
    res.add(first);
    for(int i=1;i<num.length;i++)
    {
        ArrayList<ArrayList<Integer>> newRes = new ArrayList<ArrayList<Integer>>();
        for(int j=0;j<res.size();j++)
        {
            ArrayList<Integer> cur = res.get(j);
            for(int k=0;k<cur.size()+1;k++)
            {
                ArrayList<Integer> item = new ArrayList<Integer>(cur);
                item.add(k,num[i]);
                newRes.add(item);
            }
        }
        res = newRes;
    }
    return res;   
}
上面的代码有时候乍一看可能觉得只有两层循环,时间复杂度是不是O(n^2)之类的。事实上肯定不是的,因为注意第二层循环是对于res进行遍历的,而res是一直在以平方量级增长的,所以总的时间复杂度还是会是指数量级以上的。
这种NP问题的求解在LeetCode中非常常见,类似的有N-QueensSudoku SolverCombination SumCombinations,不过思路差不多,掌握套路就不难了。这道题还有一个扩展就是如果元素集合中会出现重复,那么意味着我们需要跳过一些重复元素,具体的细节可以参见Permutations II

没有评论:

发表评论