python 回溯法 子集树模板 系列 —— 8、图的遍历

问题

一个图:
A --> B
A --> C
B --> C
B --> D
B --> E
C --> A
C --> D
D --> C
E --> F
F --> C
F --> D

从图中的一个节点E出发,不重复地经过所有其它节点后,回到出发节点E,称为一条路径。请找出所有可能的路径。

分析

将这个图可视化如下:

本问题涉及到图,那首先要考虑图用那种存储结构表示。邻接矩阵、邻接表、...都不太熟。

百度一下,在这里发现了一个最爱。这是网上找到一种最简洁的邻接表表示方式。

接下来对问题本身进行分析:

显然,问题的解的长度是固定的,亦即所有的路径长度都是固定的:n(不回到出发节点) 或 n+1(回到出发节点)

每个节点,都有各自的邻接节点。

对某个节点来说,它的所有邻接节点,可以看作这个节点的状态空间。遍历其状态空间,剪枝,深度优先递归到下一个节点。搞定!

至此,很明显套用回溯法子集树模板。

代码

'''
图的遍历

从一个节点出发,不重复地经过所有其它节点后,回到出发节点。找出所有的路径
'''

# 用邻接表表示图
n = 6  # 节点数
a,b,c,d,e,f = range(n) # 节点名称
graph = [
    {b,c},
    {c,d,e},
    {a,d},
    {c},
    {f},
    {c,d}
]

x = [0]*(n+1)  # 一个解(n+1元数组,长度固定)
X = []         # 一组解

# 冲突检测
def conflict(k):
    global n,graph,x

    # 第k个节点,是否前面已经走过
    if k < n and x[k] in x[:k]:
        return True

    # 回到出发节点
    if k == n and x[k] != x[0]:
        return True

    return False # 无冲突

# 图的遍历
def dfs(k): # 到达(解x的)第k个节点
    global n,a,b,c,d,e,f,graph,x,X

    if k > n: # 解的长度超出,已走遍n+1个节点 (若不回到出发节点,则 k==n)
        print(x)
        #X.append(x[:])
    else:
        for node in graph[x[k-1]]: # 遍历节点x[k]的邻接节点(x[k]的所有状态)
            x[k] = node
            if not conflict(k): # 剪枝
                dfs(k+1)

# 测试
x[0] = e # 出发节点
dfs(1)   # 开始处理解x中的第2个节点

效果图

时间: 2024-08-01 04:32:24

python 回溯法 子集树模板 系列 —— 8、图的遍历的相关文章

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

问题 图的m-着色判定问题 给定无向连通图G和m种不同的颜色.用这些颜色为图G的各顶点着色,每个顶点着一种颜色,是否有一种着色法使G中任意相邻的2个顶点着不同颜色? 图的m-着色优化问题 若一个图最少需要m种颜色才能使图中任意相邻的2个顶点着不同颜色,则称这个数m为该图的色数.求一个图的最小色数m的问题称为m-着色优化问题. 分析 解的长度是固定的,n.若x为本问题的一个解,则x[i]表示第i个节点的涂色编号. 可以将m种颜色看作每个节点的状态空间.每到一个节点,遍历所有颜色,剪枝,回溯. 不难

python 回溯法 子集树模板 系列 —— 3、0-1背包问题

问题 给定N个物品和一个背包.物品i的重量是Wi,其价值位Vi ,背包的容量为C.问应该如何选择装入背包的物品,使得放入背包的物品的总价值为最大? 分析 显然,放入背包的物品,是N个物品的所有子集的其中之一.N个物品中每一个物品,都有选择.不选择两种状态.因此,只需要对每一个物品的这两种状态进行遍历. 解是一个长度固定的N元0,1数组. 套用回溯法子集树模板,做起来不要太爽!!! 代码 '''0-1背包问题''' n = 3 # 物品数量 c = 30 # 包的载重量 w = [20, 15,

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

问题 旅行商问题(Traveling Salesman Problem,TSP)是旅行商要到若干个城市旅行,各城市之间的费用是已知的,为了节省费用,旅行商决定从所在城市出发,到每个城市旅行一次后返回初始城市,问他应选择什么样的路线才能使所走的总费用最短? 分析 此问题可描述如下:G=(V,E)是带权的有向图,找到包含V中每个结点一个有向环,亦即一条周游路线,使得这个有向环上所有边成本之和最小. 这个问题与前一篇文章的区别就是,本题是带权的图.只要一点小小的修改即可. 解的长度是固定的n+1. 对

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

问题 将马放到国际象棋的8*8棋盘board上的某个方格中,马按走棋规则进行移动,走遍棋盘上的64个方格,要求每个方格进入且只进入一次,找出一种可行的方案. 分析 说明:这个图是5*5的棋盘. 图片来源:这里 类似于迷宫问题,只不过此问题的解长度固定为64 每到一格,就有[(-2,1),(-1,2),(1,2),(2,1),(2,-1),(1,-2),(-1,-2),(-2,-1)]顺时针8个方向可以选择. 走到一格称为走了一步,把每一步看作元素,8个方向看作这一步的状态空间. 套用回溯法子集树

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

问题 某楼梯有n层台阶,每步只能走1级台阶,或2级台阶.从下向上爬楼梯,有多少种爬法? 分析 这个问题之前用分治法解决过.但是,这里我要用回溯法子集树模板解决它. 祭出元素-状态空间分析大法:每一步是一个元素,可走的步数[1,2]就是其状态空间.不难看出,元素不固定,状态空间固定. 直接上代码. 代码 '''爬楼梯''' n = 7 # 楼梯阶数 x = [] # 一个解(长度不固定,1-2数组,表示该步走的台阶数) X = [] # 一组解 # 冲突检测 def conflict(k): gl

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

问题 输入 第1行:字符串A 第2行:字符串B (A,B的长度 <= 1000) 输出 输出最长的子序列,如果有多个,随意输出1个. 输入示例 belong cnblogs 输出示例 blog 分析 既然打算套用回溯法子集树模板,那就要祭出元素-状态空间分析大法. 以长度较小的字符串中的字符作为元素,以长度较大的字符串中的字符作为状态空间,对每一个元素,遍历它的状态空间,其它的事情交给剪枝函数!!! 解x的长度不固定,xi表示字符串b中的序号. 在处理每一个元素时,如果没有一个状态被选择(cnb

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

作者:hhh5460 时间:2017年6月3日 用回溯法子集树模板解决了这么多问题,这里总结一下使用回溯法子集树模板的步骤: 1.确定元素及其状态空间(精髓) 对每一个元素,遍历它的状态空间,其它的事情交给剪枝函数!!!(正是这一点,使得它无愧于"通用解题法"这个称号!) 2.确定解的编码及解的长度是否固定 若解的长度固定,那么x[k] = i 若解的长度不固定,那么x.append(i) ... x.pop(i) 3.确定是求最优解,任一解,还是全部解 如果是求最优解,额外增加两个全

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

问题 给定 n 个作业,每一个作业都有两项子任务需要分别在两台机器上完成.每一个作业必须先由机器1 处理,然后由机器2处理. 试设计一个算法找出完成这n个任务的最佳调度,使其机器2完成各作业时间之和达到最小. 分析: 看一个具体的例子: tji 机器1 机器2 作业1 2 1 作业2 3 1 作业3 2 3 最优调度顺序:1 3 2 处理时间:18 这3个作业的6种可能的调度方案是1,2,3:1,3,2:2,1,3:2,3,1:3,1,2:3,2,1: 它们所相应的完成时间和分别是19,18,2

python 回溯法 子集树模板 系列 —— 2、迷宫问题

问题 给定一个迷宫,入口已知.问是否有路径从入口到出口,若有则输出一条这样的路径.注意移动可以从上.下.左.右.上左.上右.下左.下右八个方向进行.迷宫输入0表示可走,输入1表示墙.为方便起见,用1将迷宫围起来避免边界问题. 分析 考虑到左.右是相对的,因此修改为:北.东北.东.东南.南.西南.西.西北八个方向.在任意一格内,有8个方向可以选择,亦即8种状态可选.因此从入口格子开始,每进入一格都要遍历这8种状态. 显然,可以套用回溯法的子集树模板. 注意,解的长度是不固定的. 图片来源:点我 代