回溯算法-排列


文章目录

  • 回溯算法-排列
  • 一、排列
  • 二、子集
  • 三、排序与双指针
  • 1、三数之和
  • 2、最接近的三数之和



一、排列

给定一个 没有重复数字的序列[1,2,3,4],返回下面形式所有可能的全排列
[[1,2,3,4],…,]
方法1:使用python中的库方法,得到序列所有可能的排列(返回的是元组)然后转换成列表增加在一个新的列表中。

from itertools import permutations
def permute(arr):
	ret = []
	mm = permutations(arr)
	for m in mm:
		ret.append(list(m))
	return ret

方法2:使用回溯法,待排列序列[1,2,3,4]可以看成是1+permute([2,3,4] ,同理还可以把1依

次换成2,3,4,对于子序列[2,3,4]同样分解。求解的时候是将它们分解了,但是需要注意函数

返回的需要是一个组合在一起的序列,因此需要一个列表把每次分解的元素加入。根据代码

更好的理解

def permute(arr):
	n = len(arr)
	ret = []
	def add(arr,temp):
		# 分解为空的时候,此时temp里面会记录依次被分解掉的元素。
		if not arr:
			ret.append(temp)
			return 
		# 用第二个参数temp记录分解掉的元素,当分解到空集时候,用temp可以用来回溯效果。
		for i in range(n):
			add(arr[:i]+add[i+1:],temp+[arr[i]])
	add(arr,[])
	return ret

注意:对于一个字符串排列的情况,可以给上述的add(arr,’’)函数传入一个空字符串

二、子集

给你一个整数数组 nums =[1,2,3],数组中的元素 互不相同 。返回该数组所有可能的子集(幂集)。

解集不能包含重复的子集。你可以按 任意顺序返回解集:[[],[1],[2],[1,2],[3],[1,3],[2,3],[1,2,3]]。

分析:这个问题是每次分解出来的都需要立刻加入到解集中,对于任一元素,都可以与其他元素组成一个子集。这个需要记录每个元素的下标,不是一个组合问题,需要一个一个的进行分解

def subSet(arr):
	n = len(arr)
	ret = []
	def add(index,temp)
		# 每次进来就要加入
		ret.append(temp)
		for i in range(n):
			# 需要的是记录下标位置,依次加入的temp列表中。
			add(i+1, temp + [arr[i]]
	add(0,[])
	return ret

三、排序与双指针

1、三数之和

给你一个包含 n 个整数的数组arr,判断arr 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?请你找出所有和为 0 且不重复的三元组。
①三层循环,可以找到和为零的组合,但是有可能会产生重复。
②针对排序数组,看成b+c=-a问题,依次遍历赋值给a,然后通过两个首尾指针调整大小。

def three_sum(arr):
	if not arr:
		return []
	ret = []
	arr.sort()
	n = len(arr)
	for i in range(n):
		# 去重 ,保证和上一次枚举的不一样
		if i > 0 and arr[i] == arr[i-1] :
			continue
		target = -arr[i]
		# 固定第二个数,但是对于第二个每个后面的数,第三个数不是从最后开始的,就从上一次那个位置开始即可,但是对于不同的i值,第三个数都要从后面开始
		k = n - 1
		for j in range(i,n):
			if j > 1+i and arr[j] == arr[j-1]:
				continue
			while k > j and arr[j] + arr[k] > target:
				k -= 1
			# 找到一个就跳出
			if k == j: # 当前的j和后面的j都不可能了
				break
			elif arr[j] + arr[k] == target:
				ret.append([arr[i],arr[j],arr[k]])
	return ret

2、最接近的三数之和

给定一个包括 n 个整数的数组 nums 和 一个目标值 target。找出 nums 中的三个整数,使得它们的和与 target 最接近。返回这三个数的和。假定每组输入只存在唯一答案。
同样的道理通过排序和双指针,遍历第一个值,移动后面两个指针并每次记录最接近的那个和值。

def three_sum_closest(nums,target):
	n = len(nums)
	best_sum = float(1e4)
	nums.sort()
	for i in range(n):
		# 因为第一个是相当于固定一个一个遍历,所以相同的就没必要再进行下面的循环了
		if i > 0 and nums[i] == num[i-1]:
			continue
		# 同样,后面的两个数每次都要‘从头’开始
		j,k = i+1, n-1
		while j < k:
			s = nums[i] + nums[j] + nums[k]
			if s == target:
				return s
			# 不和目标相等就要记录下来
			if abs(s - target) <= abs(best_sum - target):
				best_sum = s
			if s > target :
				k -= 1
				# 这个时候需要去重,k已经减过一次,当前和上次比较
				while k > j and nums[k] == nums[k+1]:
					k -= 1
			else:
				j += 1 
				while j < k and nums[j] == nums[j-1]:
					j += 1
	 return best_sum