温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

【Python | 边学边敲边记】第二次:深度&&广度优先算法

发布时间:2020-08-11 10:40:38 阅读:203 作者:极简XksA 栏目:编程语言

【Python | 边学边敲边记】第二次:深度&&广度优先算法

一、前言

以后尽量每天更新一篇,也是自己的一个学习打卡!加油!今天给大家分享的是,Python里深度/广度优先算法介绍及实现。

二、深度、广度优先算法简介

1.深度优先搜索(DepthFirstSearch)

深度优先搜索的主要特征就是,假设一个顶点有不少相邻顶点,当我们搜索到该顶点,我们对于它的相邻顶点并不是现在就对所有都进行搜索,而是对一个顶点继续往后搜索,直到某个顶点,他周围的相邻顶点都已经被访问过了,这时他就可以返回,对它来的那个顶点的其余顶点进行搜索。
深度优先搜索的实现可以利用递归很简单地实现。

2.广度优先搜索(BreadthFirstSearch)

广度优先搜索相对于深度优先搜索侧重点不一样,深度优先好比是一个人走迷宫,一次只能选定一条路走下去,而广度优先搜索好比是一次能够有任意多个人,一次就走到和一个顶点相连的所有未访问过的顶点,然后再从这些顶点出发,继续这个过程。

具体实现的时候我们使用先进先出队列来实现这个过程:

  1. 首先将起点加入队列,然后将其出队,把和起点相连的顶点加入队列,

  2. 将队首元素v出队并标记他

  3. 将和v相连的未标记的元素加入队列,然后继续执行步骤2直到队列为空

广度优先搜索的一个重要作用就是它能够找出最短路径,这个很好理解,因为广度优先搜索相当于每次从一个起点开始向所有可能的方向走一步,那么第一次到达目的地的这个路径一定是最短的,而到达了之后由于这个点已经被访问过而不会再被访问,这个路径就不会被更改了。

三、看代码,边学边敲边记深度优先、广度优先算法

1.遍历二叉树图形

【Python | 边学边敲边记】第二次:深度&&广度优先算法

2.深度优先、广度优先遍历模型

 1


'''


 2

date : 2018.7.29


 3

author : 极简XksA


 4

goal : 深度/广度优先算法


 5

'''



 6



 7


# 深度优先: 根左右 遍历



 8


# 广度优先: 层次遍历,一层一层遍历



 9



10


# 深度优先: 根左右 遍历 (递归实现)



11



def

 

depth_tree


(tree_node)

:



12

    

if

 tree_node 

is

 

not

 

None

:


13

        print(tree_node._data)


14

        

if

 tree_node._left 

is

 

not

 

None

:


15

            

return

 depth_tree(tree_node._left)  

# 递归遍历



16

        

if

 tree_node._right 

is

 

not

 

None

:


17

            

return

 depth_tree(tree_node._right)  

# 递归遍历



18



19


# 广度优先: 层次遍历,一层一层遍历(队列实现)



20



def

 

level_queue


(root)

:



21

    

if

 root 

is

 

None

:


22

        

return



23

    my_queue = []


24

    node = root


25

    my_queue.append(node)  

# 根结点入队列



26

    

while

 my_queue:


27

        node = my_queue.pop(


) 

# 出队列



28

        print(node.elem)   

# 访问结点



29

        

if

 node.lchild 

is

 

not

 

None

:


30

            my_queue.append(node.lchild)    

# 入队列



31

        

if

 node.rchild 

is

 

not

 

None

:


32

            my_queue.append(node.rchild)    

# 入队列



3.数据结构设计、遍历代码

方法一:列表法


 1


# 树的数据结构设计



 2


# 1.列表法



 3


# 简述:列表里包含三个元素:根结点、左结点、右结点



 4

my_Tree = [


 5

    

'D'# 根结点



 6

    [

'B'

,


 7

     [

'F'

,[],[]],


 8

     [

'G'

,[

'E'

,[],[]],[]]


 9

     ],  

# 左子树



10

    [

'C'

,


11

     [],


12

     [

'A'

,[

'H'

,[],[]],[]]


13

     ]   

# 右子树



14

]


15



16


# 列表操作函数



17


# pop() 函数用于移除列表中的一个元素(默认最后一个元素),并且返回该元素的值。



18


# insert() 函数用于将指定对象插入列表的指定位置,没有返回值。



19



20


# 深度优先: 根左右 遍历 (递归实现)



21



def

 

depth_tree


(tree_node)

:



22

    

if

 tree_node:


23

        print(tree_node[


])


24

        

# 访问左子树



25

        

if

 tree_node[

1

]:


26

            depth_tree(tree_node[

1

])  

# 递归遍历



27

        

# 访问右子树



28

        

if

 tree_node[

2

]:


29

            depth_tree(tree_node[

2

])  

# 递归遍历        



30

depth_tree(my_Tree)


31


# result:



32


#            D B F G E C A H



33



34


# 广度优先: 层次遍历,一层一层遍历(队列实现)



35



def

 

level_queue


(root)

:



36

    

if

 

not

 root:


37

        

return



38

    my_queue = []


39

    node = root


40

    my_queue.append(node)  

# 根结点入队列



41

    

while

 my_queue:


42

        node = my_queue.pop(


) 

# 出队列



43

        print(node[


])   

# 访问结点



44

        

if

 node[

1

]:


45

            my_queue.append(node[

1

])    

# 入队列



46

        

if

 node[

2

]:


47

            my_queue.append(node[

2

])    

# 入队列       



48

level_queue(my_Tree)


49


# result :



50


#           D B C F G A E H



方法二:构造类节点法


 1


#2.构造类



 2


# Tree类,类变量root 为根结点,为str类型



 3


# 类变量right/left 为左右节点,为Tree型,默认为空



 4



class

 

Tree

:



 5

    root = 

''



 6

    right = 

None



 7

    left = 

None



 8

    

# 初始化类



 9

    


def

 

__init__


(self,node)

:



10

        self.root = node


11



12

    


def

 

set_root


(self,node)

:



13

        self.root = node


14



15

    


def

 

get_root


(self)

:



16

        

return

 self.root


17



18


# 初始化树



19


# 设置所有根结点



20

a = Tree(

'A'

)


21

b = Tree(

'B'

)


22

c = Tree(

'C'

)


23

d = Tree(

'D'

)


24

e = Tree(

'E'

)


25

f = Tree(

'F'

)


26

g = Tree(

'G'

)


27

h = Tree(

'H'

)


28


# 设置节点之间联系,生成树



29

a.left = h


30

b.left = f


31

b.right = g


32

c.right = a


33

d.left = b


34

d.right = c


35

g.left = e


36



37


# 深度优先: 根左右 遍历 (递归实现)



38



def

 

depth_tree


(tree_node)

:



39

    

if

 tree_node 

is

 

not

 

None

:


40

        print(tree_node.root)


41

        

if

 tree_node.left 

is

 

not

 

None

:


42

            depth_tree(tree_node.left)  

# 递归遍历



43

        

if

 tree_node.right 

is

 

not

 

None

:


44

            depth_tree(tree_node.right)  

# 递归遍历



45

depth_tree(d)  

# 传入根节点



46


# result:



47


#            D B F G E C A H



48



49


# 广度优先: 层次遍历,一层一层遍历(队列实现)



50



def

 

level_queue


(root)

:



51

    

if

 root 

is

 

None

:


52

        

return



53

    my_queue = []


54

    node = root


55

    my_queue.append(node)  

# 根结点入队列



56

    

while

 my_queue:


57

        node = my_queue.pop(


) 

# 出队列



58

        print(node.root)   

# 访问结点



59

        

if

 node.left 

is

 

not

 

None

:


60

            my_queue.append(node.left)    

# 入队列



61

        

if

 node.right 

is

 

not

 

None

:


62

            my_queue.append(node.right)    

# 入队列



63

level_queue(d)


64


# result :



65


#           D B C F G A E H



四、后言

可能大家会好奇,深度优先算法、广度优先算法对爬虫有什么用,不用好奇,慢慢后面就会懂得了。提前透露:几乎所有网站都有首页、XXX、XXX、XXX…在每个分页下,又有很多分页,这样所有url之间的联系实际上就可以比喻成一棵树,当我们想要系统爬取时,为了不重复爬取,对这颗树的遍历就尤为重要了,这里说到的深度优先、广度优先为最常用的遍历算法。

边敲边学边做,坚持学习分享。

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI