写给妹妹的编程札记 3 - 穷举: 深度优先搜索/广度优先搜索

2024-05-24 04:58

本文主要是介绍写给妹妹的编程札记 3 - 穷举: 深度优先搜索/广度优先搜索,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        前文,我们讨论了从循环遍历到搜索基本框架,并解决了一个经典的八皇后问题。对搜索剪枝也有了一些基本的了解。 下面, 我们来看看搜索的两个最基本的策略: 深度优先搜索和广度优先搜索。

        Wikipedia上有比较简单的介绍 (英文版包含更多的参考信息)

                深度优先搜索:

                        http://zh.wikipedia.org/wiki/%E6%B7%B1%E5%BA%A6%E4%BC%98%E5%85%88%E6%90%9C%E7%B4%A2

                        http://en.wikipedia.org/wiki/Depth-first_search

                广度优先搜索:

                        http://zh.wikipedia.org/wiki/%E5%B9%BF%E5%BA%A6%E4%BC%98%E5%85%88%E6%90%9C%E7%B4%A2

                        http://en.wikipedia.org/wiki/Breadth-first_search


        首先看一个比较简单的例子, 假如我们要搜索树数据结构上的节点,找出树结构上权重为5的节点,如果存在多个权重为5的节点,返回距离根节点最近的任意一个。在这里例子中, 之所以说简单的原因在于从根节点出发, 到每个节点只有一条路径可以达到。实际场景中,搜索空间中节点之间关系可能是复杂的,两个节点之间有多条可达路径。


        下面首先看深度优先搜索,怎么进行搜索?

                 第一步:搜索根节点 (A, 1), 权重为1, 并不等于5

                 第二步:任意挑选一个子节点继续 (深度+1), 访问(E, 4), 权重为4, 并不等于5

                 第三步:任意挑选一个当前节点的子节点继续 (深度+1)  - 访问(C, 3)

                              这里就是体现了深度优先, 虽然根节点(A, 1)的子节点(B, 5), (F, 8) 还没有遍历, 但我们优先往纵深方向遍历

                              (C, 3) 的权重为3, 也不等于5

                 第四步:任意挑选一个当前节点的子节点继续(深度+1) ? 

                              不过,我们发现(C, 3) 没有”更深“的子节点。 怎么办? 这个时候,我们需要回溯, 也就是访问上一个节点的其他子节点。

                              (C, 3)的上个节点是什么?我们需要保存这个信息,后进先出的栈结构适合这个需求。 因为回溯的时候,最先需要访问的是最后压入栈的节点。

                              当我们知道(C, 3)的上一个节点是(E, 4)后, 我们在第四步,尝试访问(E, 4)尚未访问过的节点(D, 5)


                              节点(D, 5)的权重等于5。 我们找到一个候选! 

                              这个时候,我们能够停止了吗? 如果题目要求只要找到任意一个权重等于5的话,我们已经找到了。

                              但,由于题目要求如果多个节点权重相等,需要返回距离根节点比较近的。 所以我们还需要继续遍历下去......

                  ......


        我们尝试结合栈数据结构,重新整理一下上面的搜索过程:

                第一步:判断根节点(A, 1),  权重不等于5

                             把根节点压入栈, 栈中包含元素[ (A, 1)

                第二步:任意挑选栈顶元素(A, 1)的尚未访问过”的一个子节点(E, 4)进行访问

                             判断节点(E, 4)的权重,权重不等于5

                             把节点(E, 4)压入栈,栈中包含元素[ (A, 1), (E, 4)

                第三步:任意挑选栈顶元素(E, 4)的尚未访问过”的一个子节点(C, 3)进行访问

                             判断节点(C, 3)的权重,权重不等于5

                             把节点(C, 3)压入栈,栈中包含元素[ (A, 1), (E, 4), (C, 3)

                第四步:任意挑选栈顶元素(C, 3)的尚未访问过”的一个子节点进行访问, 

                             但(C, 3)没有子节点, 把节点(C, 3)弹出栈,栈中包含元素[ (A, 1), (E, 4)

                第五步:任意挑选栈顶元素(E, 4)的尚未访问过”的一个子节点(D, 5)进行访问, 

                             判断节点(D, 5)的权重,权重等于5, 记录下一个解

                             把节点(D, 5)压入栈,栈中包含元素[ (A, 1), (E, 4), (D, 5)

                第六步:任意挑选栈顶元素(D, 5)的尚未访问过”的一个子节点进行访问, 

                             但(D, 5)没有子节点, 把节点(D, 5)弹出栈,栈中包含元素[ (A, 1), (E, 4)

                第六步:任意挑选栈顶元素(E, 4)的尚未访问过”的一个子节点进行访问, 

                             但(E, 4)没有”尚未访问过的“子节点, 把节点(E, 4)弹出栈,栈中包含元素[ (A, 1)

                ......

                一直进行到栈空为止


        可以看到上面的每一步操作,已经比较”机械“, 类似于《写给妹妹的编程札记 1 - 穷举: 从循环到递归》的转化,不难写出搜索程序。虽然上面描述的时候, 我们显式定义了栈数据结构,但实际程序中,有时也可以直接利用递归程序本身使用的系统调用栈。





        接下来,我们再来看广度优先搜索,怎么进行的?

                 第一步:搜索根节点 (A, 1), 权重为1, 并不等于5

                 第二步:遍历根节点的所有子节点

                                 这里就是体现了广度优先,优先遍历当前看到的所有子节点

                                 2.a 访问节点(E, 4), 权重为4, 并不等于5

                                 2.b 访问节点(B, 5), 权重为5, 等于5!

                                 2.c 访问节点(F, 8), 权重为8, 并不等于5


                                 这个时候,我们已经找到一个解,节点(B, 5), 是否应该结束呢?

                                 对于当前问题, 我们可以结束了。 因为不可能还有其他权重等于5的节点,距离根节点比(B, 5)还要近。


         如果题目要求我们找到所有权重为5的节点,那么,我们还需要继续,直到遍历完所有节点。

                 第三步:在遍历完所有根节点的子节点后, 下一步,我们应该是要遍历根节点子节点的子节点。

                                 我们怎么知道根节点的子节点是什么呢? 需要一个额外的数据结构存储起来。

                                 先进先出的队列适合这个要求,因为我们按访问过的子节点顺序,遍历这些子节点的子节点

                                 首先,遍历根节点的第一个子节点(E, 4)的所有子节点:

                                 3.a  访问节点(C, 3), 权重为3, 并不等于5

                                 3.b  访问节点(D, 5), 权重为5, 等于5!又找到一个解。

                  ......


        类似地,我们尝试结合队列数据结构,重新整理一下上面的搜索过程:

               第一步:访问根节点(A, 1),  权重不等于5

                               把根节点插入队列, 队列中包含元素 <(A, 1)>

               第二步:遍历队首节点(A,1)的所有“尚未访问过”子节点

                               2.a 访问节点(E, 4), 权重为4, 并不等于5

                                      把节点(E, 4)插入队列, 队列中包含元素<(A, 1), (E, 4)>

                               2.b 访问节点(B, 5), 权重为5, 等于5!
                                      把节点(B, 5)插入队列, 队列中包含元素<(A, 1), (E, 4), (B, 5)>
                               2.c 访问节点(F, 8), 权重为8, 并不等于5
                                      把节点(F, 8)插入队列, 队列中包含元素<(A, 1), (E, 4), (B, 5), (F, 8)>

                               到此为止,队首节点(A, 1)的所有子节点已经遍历完,可以把队首元素从队列中删除, 此时队列中包含元素<(E, 4), (B, 5), (F, 8)>

                               队列数据结构,始终保存着已经被遍历,但子节点尚未被遍历的所有节点的集合。

                第三步:遍历队首节点(E,4)的所有“尚未访问过”子节点

                                ......

                一直进行到队列为空为止。


        深度优先搜索和广度优先搜索的基本思想,如上面描述这样,非常直接和容易理解。简单的比较可以给我们一个印象:

        1. 一般如果需要找出所有解, 深度优先搜索会比广度优先搜索好。 虽然两者时间复杂度上差不多,但广度优先搜索需要比较大的空间来存储队列。

        2. 求最优解时, 一般可以优先考虑广度优先搜索。深度优先搜索一般需要遍历所有可能,而广度优先可以通过设计使得寻找到的第一个就是最优的。


        深度优先搜索和广度优先搜索的取舍还是需要具体问题具体分析。 她们是非常基本的框架,在这些基本框架的基础上, 有很多变种,技巧,技术和演化。如: A*, alpha-beta剪枝, 爬山法,beam search等等等等。 有些巧妙的算法,也充分利用和吸收了这些基本搜索策略。如,有向图的强连通分支可以通过两次深度优先遍历得到, 单源最短路的Dijkstra算法跟广度优先算法类似等等。




        

这篇关于写给妹妹的编程札记 3 - 穷举: 深度优先搜索/广度优先搜索的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/997437

相关文章

零基础STM32单片机编程入门(一)初识STM32单片机

文章目录 一.概要二.单片机型号命名规则三.STM32F103系统架构四.STM32F103C8T6单片机启动流程五.STM32F103C8T6单片机主要外设资源六.编程过程中芯片数据手册的作用1.单片机外设资源情况2.STM32单片机内部框图3.STM32单片机管脚图4.STM32单片机每个管脚可配功能5.单片机功耗数据6.FALSH编程时间,擦写次数7.I/O高低电平电压表格8.外设接口

16.Spring前世今生与Spring编程思想

1.1.课程目标 1、通过对本章内容的学习,可以掌握Spring的基本架构及各子模块之间的依赖关系。 2、 了解Spring的发展历史,启发思维。 3、 对 Spring形成一个整体的认识,为之后的深入学习做铺垫。 4、 通过对本章内容的学习,可以了解Spring版本升级的规律,从而应用到自己的系统升级版本命名。 5、Spring编程思想总结。 1.2.内容定位 Spring使用经验

好书推荐《深度学习入门 基于Python的理论与实现》

如果你对Python有一定的了解,想对深度学习的基本概念和工作原理有一个透彻的理解,想利用Python编写出简单的深度学习程序,那么这本书绝对是最佳的入门教程,理由如下:     (1)撰写者是一名日本普通的AI工作者,主要记录了他在深度学习中的笔记,这本书站在学习者的角度考虑,秉承“解剖”深度学习的底层技术,不使用任何现有的深度学习框架、尽可能仅使用基本的数学知识和Python库。从零创建一个

【文末附gpt升级秘笈】腾讯元宝AI搜索解析能力升级:千万字超长文处理的新里程碑

腾讯元宝AI搜索解析能力升级:千万字超长文处理的新里程碑 一、引言 随着人工智能技术的飞速发展,自然语言处理(NLP)和机器学习(ML)在各行各业的应用日益广泛。其中,AI搜索解析能力作为信息检索和知识抽取的核心技术,受到了广泛的关注和研究。腾讯作为互联网行业的领军企业,其在AI领域的探索和创新一直走在前列。近日,腾讯旗下的AI大模型应用——腾讯元宝,迎来了1.1.7版本的升级,新版本在AI搜

IPython小白教程:提升你的Python交互式编程技巧,通俗易懂!

IPython是一个增强的Python交互式shell,它提供了丰富的功能和便捷的交互方式,使得Python开发和数据分析工作更加高效。本文将详细介绍IPython的基本概念、使用方法、主要作用以及注意事项。 一、IPython简介 1. IPython的起源 IPython由Fernando Pérez于2001年创建,旨在提供一个更高效的Python交互式编程环境。 2. IPyt

从《深入设计模式》一书中学到的编程智慧

软件设计原则   优秀设计的特征   在开始学习实际的模式前,让我们来看看软件架构的设计过程,了解一下需要达成目标与需要尽量避免的陷阱。 代码复用 无论是开发何种软件产品,成本和时间都最重要的两个维度。较短的开发时间意味着可比竞争对手更早进入市场; 较低的开发成本意味着能够留出更多营销资金,因此能更广泛地覆盖潜在客户。 代码复用是减少开发成本时最常用的方式之一。其意图

【图像识别系统】昆虫识别Python+卷积神经网络算法+人工智能+深度学习+机器学习+TensorFlow+ResNet50

一、介绍 昆虫识别系统,使用Python作为主要开发语言。通过TensorFlow搭建ResNet50卷积神经网络算法(CNN)模型。通过对10种常见的昆虫图片数据集(‘蜜蜂’, ‘甲虫’, ‘蝴蝶’, ‘蝉’, ‘蜻蜓’, ‘蚱蜢’, ‘蛾’, ‘蝎子’, ‘蜗牛’, ‘蜘蛛’)进行训练,得到一个识别精度较高的H5格式模型文件,然后使用Django搭建Web网页端可视化操作界面,实现用户上传一

基于深度学习的轮廓检测

基于深度学习的轮廓检测 轮廓检测是计算机视觉中的一项关键任务,旨在识别图像中物体的边界或轮廓。传统的轮廓检测方法如Canny边缘检测和Sobel算子依赖于梯度计算和阈值分割。而基于深度学习的方法通过训练神经网络来自动学习图像中的轮廓特征,能够在复杂背景和噪声条件下实现更精确和鲁棒的检测效果。 深度学习在轮廓检测中的优势 自动特征提取:深度学习模型能够自动从数据中学习多层次的特征表示,而不需要

Java并发编程—阻塞队列源码分析

在前面几篇文章中,我们讨论了同步容器(Hashtable、Vector),也讨论了并发容器(ConcurrentHashMap、CopyOnWriteArrayList),这些工具都为我们编写多线程程序提供了很大的方便。今天我们来讨论另外一类容器:阻塞队列。   在前面我们接触的队列都是非阻塞队列,比如PriorityQueue、LinkedList(LinkedList是双向链表,它实现了D