一直不是太理解回溯法,这几天集中学习了一下,记录如下。

回溯法有“通用的解题法”之称。

1.定义:

  • 也叫试探法,它是一种系统地搜索问题的解的方法。

2.基本思想:

  • 从一条路往前走,能进则进,不能进则退回来,换一条路再试。

3.一般步骤:

  • 定义一个解空间(子集树、排列树二选一)
  • 利用适于搜索的方法组织解空间。
  • 利用深度优先法搜索解空间。
  • 利用剪枝函数避免移动到不可能产生解的子空间。

4.约束函数:

  • 是否满足显约束(存在)

5.限界函数:

  • 是否满足隐约束(最优)

6.子集树模板

遍历子集树,时间复杂度 O(2^n)

如果解的长度是不固定的,那么解和元素顺序无关,即可以从中选择0个或多个。例如:子集,迷宫,...

如果解的长度是固定的,那么解和元素顺序有关,即每个元素有一个对应的状态。例如:子集,8皇后,...

解空间的个数指数级别的,为2^n,可以用子集树来表示所有的解

适用于:幂集、子集和、0-1背包、装载、8皇后、迷宫、...

a.子集树模板递归版

'''求集合{1, 2, 3, 4}的所有子集''' n = 4
#a = ['a','b','c','d']
a = [1, 2, 3, 4]
x = [] # 一个解(n元0-1数组)
X = [] # 一组解 # 冲突检测:无
def conflict(k):
global n, x, X, a return False # 无冲突 # 一个例子
# 冲突检测:奇偶性相同,且和小于8的子集
def conflict2(k):
global n, x, X, a if k==0:
return False # 根据部分解,构造部分集
s = [y[0] for y in filter(lambda s:s[1]!=0, zip(a[:k+1],x[:k+1]))]
if len(s)==0:
return False
if 0 < sum(map(lambda y:y%2, s)) < len(s) or sum(s) >= 8: # 只比较 x[k] 与 x[k-1] 奇偶是否相间
return True return False # 无冲突 # 子集树递归模板
def subsets(k): # 到达第k个元素
global n, x, X if k >= n: # 超出最尾的元素
#print(x)
X.append(x[:]) # 保存(一个解)
else:
for i in [1, 0]: # 遍历元素 a[k] 的两种选择状态:1-选择,0-不选
x.append(i)
if not conflict2(k): # 剪枝
subsets(k+1)
x.pop() # 回溯 # 根据一个解x,构造一个子集
def get_a_subset(x):
global a return [y[0] for y in filter(lambda s:s[1]!=0, zip(a,x))] # 根据一组解X, 构造一组子集
def get_all_subset(X):
return [get_a_subset(x) for x in X] # 测试
subsets(0) # 查看第3个解,及对应的子集
#print(X[2])
#print(get_a_subset(X[2])) print(get_all_subset(X))
b.子集树模板非递归版

7.排列树模板

遍历排列树,时间复杂度O(n!)

解空间是由n个元素的排列形成,也就是说n个元素的每一个排列都是解空间中的一个元素,那么,最后解空间的组织形式是排列树

适用于:n个元素全排列、旅行商、...

a.排列树模板递归版

'''求[1,2,3,4]的全排列''' n = 4
x = [1,2,3,4] # 一个解
X = [] # 一组解 # 冲突检测:无
def conflict(k):
global n, x, X return False # 无冲突 # 一个例子
# 冲突检测:元素奇偶相间的排列
def conflict2(k):
global n, x, X if k==0: # 第一个元素,肯定无冲突
return False if x[k-1] % 2 == x[k] % 2: # 只比较 x[k] 与 x[k-1] 奇偶是否相同
return True return False # 无冲突 # 排列树递归模板
def backkrak(k): # 到达第k个位置
global n, x, X if k >= n: # 超出最尾的位置
print(x)
#X.append(x[:]) # 注意x[:]
else:
for i in range(k, n): # 遍历后面第 k~n-1 的位置
x[k], x[i] = x[i], x[k]
if not conflict2(k): # 剪枝
backkrak(k+1)
x[i], x[k] = x[k], x[i] # 回溯 # 测试
backkrak(0)
b.排列树模板非递归版

python 回溯法 记录的更多相关文章

  1. python 回溯法 子集树模板 系列 —— 19、野人与传教士问题

    问题 在河的左岸有N个传教士.N个野人和一条船,传教士们想用这条船把所有人都运过河去,但有以下条件限制: (1)修道士和野人都会划船,但船每次最多只能运M个人: (2)在任何岸边以及船上,野人数目都不 ...

  2. python 回溯法 子集树模板 系列 —— 18、马踏棋盘

    问题 将马放到国际象棋的8*8棋盘board上的某个方格中,马按走棋规则进行移动,走遍棋盘上的64个方格,要求每个方格进入且只进入一次,找出一种可行的方案. 分析 说明:这个图是5*5的棋盘. 图片来 ...

  3. python 回溯法 子集树模板 系列 —— 17、找零问题

    问题 有面额10元.5元.2元.1元的硬币,数量分别为3个.5个.7个.12个.现在需要给顾客找零16元,要求硬币的个数最少,应该如何找零?或者指出该问题无解. 分析 元素--状态空间分析大法:四种面 ...

  4. python 回溯法 子集树模板 系列 —— 16、爬楼梯

    问题 某楼梯有n层台阶,每步只能走1级台阶,或2级台阶.从下向上爬楼梯,有多少种爬法? 分析 这个问题之前用分治法解决过.但是,这里我要用回溯法子集树模板解决它. 祭出元素-状态空间分析大法:每一步是 ...

  5. python 回溯法 子集树模板 系列 —— 15、总结

    作者:hhh5460 时间:2017年6月3日 用回溯法子集树模板解决了这么多问题,这里总结一下使用回溯法子集树模板的步骤: 1.确定元素及其状态空间(精髓) 对每一个元素,遍历它的状态空间,其它的事 ...

  6. python 回溯法 子集树模板 系列 —— 14、最长公共子序列(LCS)

    问题 输入 第1行:字符串A 第2行:字符串B (A,B的长度 <= 1000) 输出 输出最长的子序列,如果有多个,随意输出1个. 输入示例 belong cnblogs 输出示例 blog ...

  7. python 回溯法 子集树模板 系列 —— 13、最佳作业调度问题

    问题 给定 n 个作业,每一个作业都有两项子任务需要分别在两台机器上完成.每一个作业必须先由机器1 处理,然后由机器2处理. 试设计一个算法找出完成这n个任务的最佳调度,使其机器2完成各作业时间之和达 ...

  8. python 回溯法 子集树模板 系列 —— 10、m着色问题

    问题 图的m-着色判定问题 给定无向连通图G和m种不同的颜色.用这些颜色为图G的各顶点着色,每个顶点着一种颜色,是否有一种着色法使G中任意相邻的2个顶点着不同颜色? 图的m-着色优化问题 若一个图最少 ...

  9. python 回溯法 子集树模板 系列 —— 9、旅行商问题(TSP)

    问题 旅行商问题(Traveling Salesman Problem,TSP)是旅行商要到若干个城市旅行,各城市之间的费用是已知的,为了节省费用,旅行商决定从所在城市出发,到每个城市旅行一次后返回初 ...

随机推荐

  1. 解决 jQuery validation插件 valid()方法总是返回true的问题

    在表单元素验证方法中加入了 remote 方法,调试一直返回true,后来才知道因为是异步验证,所以才会出现此问题,解决方法就是在 remote 方法中,禁用 异步和缓存,具体代码如下 : ..... ...

  2. 使用anaconda安装pytorch的清华镜像地址

    1.安装anaconda:国内镜像网址:https://mirror.tuna.tsinghua.edu.cn/help/anaconda/下载对应系统对应python版本的anaconda版本(Li ...

  3. python常用模块之-random模块

    random模块顾名思义就是生成随机数的模块. random模块有以下常见方法: 1,打印0-1之间的任意随机浮点数,不能指定区间. print(random.random()) 2,打印随机符点数, ...

  4. SQL Server 2014 新特性——内存数据库(转载)

    目录 SQL Server 2014 新特性——内存数据库 简介: 设计目的和原因: 专业名词 In-Memory OLTP不同之处 内存优化表 内存优化表的索引 并发能力的提升 和竞争对手相比几点 ...

  5. PowerShell发送邮件(587)

    #定义邮件服务器 $smtpServer = "mail.xx.com" $smtpUser = "sender" $smtpPassword = " ...

  6. oracle like模糊查询简单用法

    like  用法介绍: 1.“_”:匹配单个任意字符 select * from bqh3 where name like '_崔'; 2.“%”:匹配0个或多个任意字符.但有三种情况如下: like ...

  7. python永久添加第三方模块,PYTHONPATH的设置

    今天用pip安装pymysql后遇到了一个问题,在PyCharm中import pymysql模块时,运行却提示我找不到pymysql mudule 我先考虑的是pymysql没有安装成功,但是cmd ...

  8. java读取txt文件的2中方法---并将内容(每一行以固定的字符分割切成2段)存到map中去

    #java读取txt文件的第一种方法 /** * 方法:readTxt * 功能:读取txt文件并把txt文件的内容---每一行作为一个字符串加入到List中去 * 参数:txt文件的地址 * 返回: ...

  9. 【转】VMware虚拟机三种网络模式超详解

    [原文]https://www.toutiao.com/i6596228488666022403/ 由于Linux目前很热门,越来越多的人在学习Linux,但是买一台服务放家里来学习,实在是很浪费.那 ...

  10. Linux运维之——每日小技巧,谈进程与线程的区别

    线程是进程中执行运算的最小单位,是进程中的一个实体,是被系统独立调度和分派的基本单位,线程自己不拥有系统资源,只拥有一点在运行中必不可少的资源,但它可与同属一个进程的其它线程共享进程所拥有的全部资源. ...