46. 全排列(Python)

更多精彩内容,请关注【力扣中等题】

题目

难度:★★★☆☆
类型:数学
方法:回溯法

题目

给定一个没有重复数字的序列,返回其所有可能的全排列。

示例

输入: [1,2,3]
输出:
[
[1,2,3],
[1,3,2],
[2,1,3],
[2,3,1],
[3,1,2],
[3,2,1]
]

解答

全排列其实可以使用python内置的permutations函数,例如求['a', 'b', 'c']的全排列,可以使用:itertools.permutations(['a', 'b', 'c'],3)快速得到。这里参考了大佬博客

方案1:回溯法

我们举个例子,以字符串列表['a', 'b', 'c']为例,我们逐个位确定全排列的所有可能。回溯法的原理在于在前n-1位元素确定的情况下,求取n位以后的全排列。本例中,首先固定第0位,就是分别将第0位与它本身及后面各位元素交换,得到3种不同的可能,在固定这一位后,在考虑第1位的可能性,将第1位与它本身及其后元素交换,有两种可能性,当前两位元素确定后,最后一位只有一种可能性。因此一共有6种可能。

  1. 将列表的第0位与第0位交换(相当于不变),此时列表变为['a', 'b', 'c'];
    1.1 将列表的第1位与第1位交换(相当于不变),得到列表['a', 'b', 'c'];
    1.2 将列表的第1位与第2位交换,得到列表['a', 'c', 'b'];

  2. 将列表的第0位与第1位交换,得到列表['b', 'a', 'c'];
    2.1 将列表的第1位与第1位交换(相当于不变),得到列表['b', 'a', 'c'];
    2.2 将列表的第1位与第2位交换,得到列表['b', 'c', 'a'];

  3. 将列表的第0位与第2位交换,得到列表['c', 'b', 'a'];
    3.1 将列表的第1位与第1位交换(相当于不变),得到列表['c', 'b', 'a'];
    3.2 将列表的第1位与第2位交换,得到列表['c', 'a', 'b']

这里需要注意的是,每次交换元素并回溯寻找后,都要将元素交换回来,保持没有交换前的状态。

class Solution:
    def permute(self, nums):
        """
        :type nums: List[int]
        :rtype: List[List[int]]
        """

        def backtrack(position, end):
            """
            Find possible results using backtrack.
            :param position:
            :param end:
            :return:
            """

            if position == end:
                res.append(nums[:])
                return

            for index in range(position, end):
                nums[index], nums[position] = nums[position], nums[index]
                backtrack(position + 1, end)
                nums[index], nums[position] = nums[position], nums[index]

        res = []
        backtrack(0, len(nums))
        return res

方案2:深度优先搜索

与回溯法类似,增加临时列表用来存储是否查看过变量。

class Solution:
    def permute(self, nums):
        """
        :type nums: List[int]
        :rtype: List[List[int]]
        """

        visit = [True for _ in range(len(nums))]
        tmp = nums[:]

        def dfs(position):
            if position == len(nums):
                res.append(tmp[:])
                return

            for index in range(0, len(nums)):
                if visit[index]:
                    tmp[position] = nums[index]
                    visit[index] = False
                    dfs(position + 1)
                    visit[index] = True

        res = []
        dfs(0)
        return res

如有疑问或建议,欢迎评论区留言~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 专业考题类型管理运行工作负责人一般作业考题内容选项A选项B选项C选项D选项E选项F正确答案 变电单选GYSZ本规程...
    小白兔去钓鱼阅读 9,095评论 0 13
  • 1.埋点是做什么的 2.如何进行埋点 3.埋点方案的设计 近期常被问到这个问题,我担心我的答案会将一些天真烂漫的孩...
    lxg阅读 2,038评论 0 1
  • 递归+交换 递归+跳过已选数字 第一种方法的图解,有空补
    Peepea阅读 598评论 0 0
  • 选择题部分 1.(),只有在发生短路事故时或者在负荷电流较大时,变流器中才会有足够的二次电流作为继电保护跳闸之用。...
    skystarwuwei阅读 13,699评论 0 7
  • 1. 关于诊断X线机准直器的作用,错误的是()。 (6.0 分) A. 显示照射野 B. 显示中心线 C. 屏蔽多...
    我们村我最帅阅读 10,899评论 0 5