Go语言二叉树数据结构的应用
来源:云海天教程
时间:2022-12-31 08:41:27 373浏览 收藏
本篇文章主要是结合我之前面试的各种经历和实战开发中遇到的问题解决经验整理的,希望这篇《Go语言二叉树数据结构的应用》对你有很大帮助!欢迎收藏,分享给更多的需要的朋友学习~
树型结构(Tree)是一种重要的非线性数据结构,它为计算机应用中出现的具有层次关系的数据提供了一种有效的表示方法,比如文件目录结构、源程序语法结构等。树的定义和基本术语
树是 n(n>=0) 个节点的有限集合 T。在任意一棵非空树中满足如下两个条件:有且仅有一个根节点(Root)。当 n>1 时,其余节点可分为 m(m>=0) 个互不相交的有限集合 T1,T2,……,Tm,其中每一个集合本身又都是一棵树,并且称为根的子树(Subtree),如下图所示。
图:树型结构
由上图可知,树的定义是递归的,树是一种递归数据结构。树的这种定义为树的递归处理带来了很大方便,本节举例中几乎所有对树的处理都采用了递归算法。
在了解树型结构时,还有几个基本概念非常重要,必须要掌握:
节点的度:树中每个节点具有的子树数,或后继节点数称为该节点的度。树的度:树中所有节点的度的最大值称为树的度。分支节点:度大于 0 的节点称为分支节点或非终端节点。叶子节点:度为 0 的节点称为叶子节点或终端节点。儿子节点:一个节点的后继称为该节点的儿子节点。父亲节点:一个节点称为其后继节点的父亲节点。子孙节点:一个节点的所有子树中的节点称为该节点的子孙节点。祖先节点:从根节点到达一个节点的路径上,通过的所有节点称为该节点的祖先节点。兄弟节点:具有同一父亲的节点相互称为兄弟节点。节点的层数:树是一种层次结构,根节点为第一层,其儿子节点为第二层,以此类推可以得到每个节点的层数。树的深度:树中节点的最大层数称为树的深度或高度。森林:0 个或多个不相交的树的集合称为森林。
二叉树简介
二叉树(Binary Tree)是一种特殊的树型结构,二叉树中每个节点至多有两棵子树,称为左子树、右子树。即二叉树中不存在度大于 2 的节点,且两棵子树有左右之分,次序不能任意颠倒。除了这些基本特征外,还有如下一些特殊的二叉树。
满二叉树:在一棵二叉树中,若第 i 层的节点数为 2i-1 时,则称此层的节点数是满的,当二叉树中的每一层都是满的,则称此二叉树为满二叉树。完全二叉树:在一棵二叉树中,除最后一层外,若其余各层都是满的,并且最后一层或者是满的,或者是在右边缺少连续若干个节点,则此二叉树被称为完全二叉树。
二叉树的链接存储结构
二叉树的存储方法有顺序存储、链接存储和线索树存储等几种存储方法,顺序存储是使用数组来完成,而链接存储和线索树存储都使用了链表来完成。本节在二叉树的应用举例中使用了链接存储法。为什么不使用线索树呢?原因前面讲了,树型结构的主要特征是递归,所以对树的所有操作都使用递归算法来实现,而线索树恰恰是使用非递归算法来加速遍历速度。节点定义
在二叉树的链接存储结构中,通常采用的方法是:每个节点设置三个域,即值域、左指针域和右指针域,其结构如下图所示。图:二叉树链接存储结构
其中 Data 表示值域,用于存储放入节点中的数据元素,Left 和 Right 分别表示左指针域和右指针域,用以分别存储左孩子和右孩子节点的指针地址。
链接存储的指针类型和节点定义如下:
type Node struct {
Left *Node
Data interface{}
Right *Node
}
接口定义
二叉树的应用处理功能主要包括:二叉树新节点的创建、初始化;二叉树的输出、度的计算、叶子节点统计等基本操作;二叉树的前序、中序、后序遍历等。针对这些功能本例共定义了三个接口:Initer、Operater 和 Order。1) Initer 接口。当为二叉树创建了一个新节点时,Initer 接口提供了方法 SetData() 可以对节点的 Data 字段进行初始化。
Initer 接口定义如下:
type Initer interface {
SetData (data interface{})
}
Operater 接口定义如下:
type Operater interface {
PrintBT()
Depth() int
LeafCount() int
}
Order 接口的定义如下:
type Order interface {
PreOrder()
InOrder()
PostOrder()
}
接口的意义也在于此,即同一个接口可以有不同的实现方法,甚至由不同的人去完成。所以,Go语言是软件工程类的高级程序设计语言。
方法的实现
通过接口的概念知道接口是方法的组合,而在定义接口时不必马上实现方法,方法可由设计者自己或交由其他人另外单独设计。本例三个接口中共有 7 个方法,在此分别介绍它们的实现算法。1) SetData 方法。SetData() 通过空接口 interface{},可以将任意类型数据赋值给二叉树节点 Node 的 Data 字段,实现二叉树对任意数据类型的存储。
SetData 方法的定义如下:
func (n *Node) SetData(data interface{}) {
n.Data = data
}
PrintBT 方法的定义如下:
func (n *Node) PrintBT() {
PrintBT(n)
}
Depth 方法的定义如下:
func (n *Node) Depth() int {
return Depth(n)
}
LeafCount 方法的定义如下:
func (n *Node) LeafCount() int {
return LeafCount(n)
}
PreOrder 方法的定义如下:
func (n *Node) PreOrder() {
PreOrder(n)
}
InOrder 方法的定义如下:
func (n *Node) InOrder() {
InOrder(n)
}
PostOrder 方法的定义如下:
func (n *Node) PostOrder() {
PostOrder(n)
}
底层函数设计
在面向对象程序设计中,上层方法的功能实现还要依赖底层函数,本例中大部分方法也是这样,现将所有底层函数一一列举如下。1) NewNode 函数。NewNode() 按照链接存储方式生成一个新的二叉树节点,参数 left 指向左指针域,参数 right 指向右指针域。
NewNode 函数的定义如下:
func NewNode(left, right *Node) *Node {
return &Node{left, nil, right}
}
PrintBT 函数的定义如下:
func PrintBT(n *Node) { if n != nil { fmt.Printf("%v", n.Data) if n.Left != nil || n.Right != nil { fmt.Printf("(") PrintBT(n.Left) if n.Right != nil { fmt.Printf(",") } PrintBT(n.Right) fmt.Printf(")") } }}3) Depth 函数。Depth() 用于计算二叉树的深度,采用递归算法:若一棵二叉树为空,则其深度为 0;否则,其深度等于左子树或右子树的最大深度加 1。
Depth 函数的定义如下:
func Depth(n *Node) int { var depleft, depright int if n == nil { return 0 } else { depleft = Depth(n.Left) depright = Depth(n.Right) if depleft > depright { return depleft + 1 } else { return depright + 1 } }}4) LeafCount 函数。LeafCount() 用于统计二叉树叶子节点数,采用递归算法:若一棵二叉树为空,则其叶子节点数为 0;若一棵二叉树的左、右子树均为空,则其叶子节点数为 1;否则叶子数等于左子树与右子树叶子总数之和。
LeafCount 函数的定义如下:
func LeafCount(n *Node) int { if n == nil { return 0 } else if (n.Left == nil) && (n.Right == nil) { return 1 } else { return (LeafCount(n.Left) + LeafCount(n.Right)) }}5) PreOrder 函数。PreOrder() 可以对二叉树进行前序遍历,采用递归算法:按照先访问根节点,再访问左子树,最后访问右子树的次序访问二叉树中的所有节点,且每个节点仅访问一次。
PreOrder 函数的定义如下:
func PreOrder(n *Node) { if n != nil { fmt.Printf("%v", n.Data) PreOrder(n.Left) PreOrder(n.Right) }}6) InOrder 函数。InOrder() 可以对二叉树进行中序遍历,采用递归算法:按照先访问左子树,再访问根节点,最后访问右子树的次序访问二叉树中的所有节点,且每个节点仅访问一次。
InOrder 函数的定义如下:
func InOrder(n *Node) { if n != nil { PreOrder(n.Left) fmt.Printf("%v", n.Data) PreOrder(n.Right) }}7) PostOrder 函数。PostOrder() 可以对二叉树进行后序遍历,采用递归算法:按照先访问左子树,再访问右子树,最后访问根节点的次序访问二叉树中的所有节点,且每个节点仅访问一次。
PostOrder 函数的定义如下:
func PostOrder(n *Node) { PreOrder(n.Left) PreOrder(n.Right) fmt.Printf("%v", n.Data)}
二叉树基本应用测试
前面介绍了二叉树链接存储结构,以及节点定义、接口定义、方法实现和底层函数设计,并在包 btree 中实现。接下来将利用这些知识实现几个二叉树的基本应用,包括二叉树的创建、基本操作和遍历。二叉树创建
二叉树的创建过程一般如下:首先调用 NewNode() 函数创建根节点,再调用 SetData() 方法初始化根节点。使用同样的方法创建左子树和右子树,并将左、右子树链接到根节点上。如果左、右子树有叶子节点,则插入叶子节点并和左、右子树建立链接。
【示例 1】二叉树的建立,本例将演示如何使用 Initer 接口实现根节点的初始化。
// 二叉树的建立package mainimport( "fmt" "btree")func main() { //创建根节点 root := NewNode(nil, nil) var it Initer it = root it.SetData("root node") //创建左子树 a := NewNode(nil, nil) a.SetData("left node") al := NewNode(nil, nil) //左叶子节点 al.SetData(100) ar := NewNode(nil, nil) //右叶子节点 ar.SetData(3.14) a.Left = al a.Right = ar //创建右子树 b := NewNode(nil, nil) b.SetData("right node") root.Left = a root.Right = b root.PrintBT()}编译并运行该程序,输出结果为:
root node (left node ( 100,3.14 ),right node)
通过输出结果可以看出,在示例中首先创建了根节点 root node,然后创建左子树 left node 和右子树 right node。左子树有两个叶子节点,左叶子的值为 int 型 "100",右叶子的值为 float 型 "3.14"。即该二叉树可以存储不同类型的值,这些都是由空接口 interface{} 实现的。二叉树基本操作
对一个已存在的二叉树的基本操作包括二叉树的输出、深度计算、叶子数统计等,在下面将演示如何使用 Operater 接口实现这些基本操作。【示例 2】二叉树的基本操作。
// 二叉树的基本操作package mainimport( "fmt" "btree")func main() { //创建二叉树 root := NewNode(nil, nil) root.SetData("root node") a := NewNode(nil, nil) a.SetData("left node") al := NewNode(nil, nil) al.SetData(100) ar := NewNode(nil, nil) ar.SetData(3.14) a.Left = al a.Right = ar b := NewNode(nil, nil) b.SetData("right node") root.Left = a root.Right = b // 使用 Operater 接口实现对二叉树的基本操作 var it Operater it = root it.PrintBT() fmt.Println() fmt.Println("The depths of the Btree is:", it.Depth()) fmt.Println("The leaf counts of the Btree is:", it.LeafCount())}编译并运行该程序,输出结果为:
root node ( left node (100,3.14),right node )
The depths of the Btree is: 3
The leaf counts of the Btree is: 3
二叉树遍历
在二叉树的一些基本应用中,常常需要在树中查找具有某种特征的节点,或者对树中全部节点逐一进行某种处理,这就是二叉树的遍历(Traversing binary tree)。即如何按某条搜索路径访问树中的每个节点,使得每个节点均能被访问一次,而且仅被访问一次。二叉树的遍历方法一般分为前序遍历、中序遍历和后序遍历,下面示例将演示如何使用 Order 接口实现二叉树的三种遍历。
【示例 3】二叉树的遍历。
// 二叉树的遍历package mainimport( "fmt" "btree")func main() { //创建二叉树 root := NewNode(nil, nil) root.SetData("root node") a := NewNode(nil, nil) a.SetData("left node") al := NewNode(nil, nil) al.SetData(100) ar := NewNode(nil, nil) ar.SetData(3.14) a.Left = al a.Right = ar b := NewNode(nil, nil) b.SetData("right node") root.Left = a root.Right = b // 使用 Order 接口实现对二叉树的基本操作 var it Order it = root it.PreOrder() //先序遍历 fmt.Println() it.InOrder() //中序遍历 fmt.Println() it.PostOrder() //后序遍历}编译并运行该程序,输出结果为:
root node left node 100 3.14 right node
left node 100 3.14 root node right node
left node 100 3.14 right node root node
今天关于《Go语言二叉树数据结构的应用》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!
-
250 收藏
-
462 收藏
-
394 收藏
-
256 收藏
-
282 收藏
-
438 收藏
-
280 收藏
-
181 收藏
-
371 收藏
-
236 收藏
-
416 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习