全排列(重复值和无重复值的情况)

本文深入解析了全排列算法,包括处理无重复数值和有重复数值的序列,提供Python实现代码,展示如何通过递归实现全排列,并处理重复元素,确保结果的唯一性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

没有重复数值:
输入
[1,2,3]
输出
[[1,2,3],[1,3,2],[2,1,3],[2,3,1],
[3,2,1],[3,1,2]]
ans.append(nums);不加nums+[]的结果如下:
[[1, 2, 3], [1, 2, 3], [1, 2, 3], [1, 2, 3],
 [1, 2, 3], [1, 2, 3]]

class Solution:
    def permute(self, nums: List[int]) -> List[List[int]]:
        ans=[]
        def dfs(nums,start):
        //当start==arr.length-1时,说明子序列的长度为1,就不用再往下分子序列了
            if start==len(nums)-1:
            一定要加nums+[](新建):这个就和nums没关系了
                ans.append(nums+[]);
                return;
            for i in range(start,len(nums)):
            //start代表的是每一个子序列的第一个位置,
            我们每一层递归的任务都只有一个:
                        //枚举该层子序列第一个位置可以取的值
                nums[start],nums[i]=nums[i],nums[start];
                //该层递归的子序列第一个位置已经确定了,
                所以又可以往下再分
                dfs(nums,start+1);
                 //把第该层子序列第一个位置的值换成另外一个值,
                 所以要交换回来
                nums[start],nums[i]=nums[i],nums[start];
                
        dfs(nums,0);
        return ans;
有重复值的:
给定一个可包含重复数字的序列,返回所有不重复的全排列。

示例:

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

class Solution:
    def permuteUnique(self, nums: List[int]) -> List[List[int]]:
        ans=[];
        # sorted(nums);
        canSwap=[False for _ in range(len(nums))];
        /*dfs全排列搜索前,先判断是否交换过,效率98%+*/
        def canSwap(nums,start,end):
            for i in range(start,end):
                if nums[i]==nums[end]:
                    return False;
            return True;
        def dfs(nums,start):
            if start==len(nums)-1:
                now=nums+[];
                if now not in ans:ans.append(now);
                rxeturn ;
            for i in range(start,len(nums)):
            /*dfs全排列搜索前,先判断是否交换过,效率98%+*/
                if canSwap(nums,start,i):
                    nums[start],nums[i]=nums[i],nums[start];
                    dfs(nums,start+1);
                    nums[start],nums[i]=nums[i],nums[start];
        dfs(nums,0);
        return ans;
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值