二叉树的二叉链表存储结构及C++实现
前言:存储二叉树的关键是如何表示结点之间的逻辑关系,也就是双亲和孩子之间的关系。在具体应用中,可能要求从任一结点能直接访问到它的孩子。
一、二叉链表
二叉树一般多采用二叉链表(binary linked list)存储,其基本思想是:令二叉树的每一个结点对应一个链表结点链表结点除了存放与二叉树结点有关的数据信息外,还要设置指示左右孩子的指针。二叉链表的结点结构如下图所示:
| lchild | data | rchild |
其中,data为数据域,存放该结点的数据信息;
lchild为左指针域,存放指向左孩子的指针,当左孩子不存在时为空指针;
rchild为右指针域,存放指向右孩子的指针,当右孩子不存在时为空指针;
可以用C++语言中的结构体类型描述二叉链表的结点,由于二叉链表的结点类型不确定,所以采用C++的模板机制。如下:
// 二叉链表的节点
template<class T>
struct BiNode
{
T data; // 数据域
BiNode<T>*lchild, *rchild; // 左右指针域
};
二、C++实现
将二叉树的二叉链表存储结构用C++的类实现。为了避免类的调用者访问BiTree类的私有变量root,在构造函数、析构函数以及遍历函数中调用了相应的私有函数。
具体代码实现如下:
1、头文件“cirqueue.h”
此头文件为队列的类实现,层序遍历要用到队列,所以自己定义了一个队列。
#pragma once
#include <iostream>
const int queueSize = ;
template<class T>
class queue
{
public:
....
T data[queueSize];
int front, rear;
....
};
2、头文件“bitree.h”
此头文件为二叉链表的类实现。
#pragma once
#include <iostream>
#include "cirqueue.h"
// 二叉链表的节点
template<class T>
struct BiNode
{
T data; // 数据域
BiNode<T>*lchild, *rchild; // 左右指针域
};
// 二叉链表类实现
template<class T>
class BiTree
{
public:
BiTree() { root = Creat(root); } // 构造函数,建立一颗二叉树
~BiTree() { Release(root); } // 析构函数,释放各节点的存储空间
void PreOrder() { PreOrder(root); } // 递归前序遍历二叉树
void InOrder() { InOrder(root); } // 递归中序遍历二叉树
void PostOrder() { PostOrder(root); } // 递归后序遍历二叉树
void LeverOrder(); // 层序遍历二叉树
private:
BiNode<T>* root; // 指向根节点的头节点
BiNode<T>* Creat(BiNode<T>* bt); // 构造函数调用
void Release(BiNode<T>* bt); // 析构函数调用
void PreOrder(BiNode<T>* bt); // 前序遍历函数调用
void InOrder(BiNode<T>* bt); // 中序遍历函数调用
void PostOrder(BiNode<T>* bt); // 后序遍历函数调用
}; template<class T>
inline void BiTree<T>::LeverOrder()
{
queue<BiNode<T>*> Q; // 定义一个队列
Q.front = Q.rear = -; // 顺序队列
if (root == NULL)
return;
Q.data[++Q.rear] = root; // 根指针入队
while (Q.front != Q.rear)
{
BiNode<T>* q = Q.data[++Q.front]; // 出队
cout << q->data;
if (q->lchild != NULL)
Q.data[++Q.rear] = q->lchild; // 左孩子入队
if (q->rchild != NULL)
Q.data[++Q.rear] = q->rchild; // 右孩子入队
} } template<class T>
inline BiNode<T>* BiTree<T>::Creat(BiNode<T>* bt)
{
T ch;
cin >> ch; // 输入结点的数据信息,假设为字符
if (ch == '#') // 建立一棵空树
bt = NULL;
else
{
bt = new BiNode<T>; // 生成一个结点,数据域为ch
bt->data = ch;
bt->lchild = Creat(bt->lchild); // 递归建立左子树
bt->rchild = Creat(bt->rchild); // 递归建立右子树
}
return bt;
} template<class T>
inline void BiTree<T>::Release(BiNode<T>* bt)
{
if (bt != NULL)
{
Release(bt->lchild); // 释放左子树
Release(bt->rchild); // 释放右子树
delete bt; // 释放根节点
}
} template<class T>
inline void BiTree<T>::PreOrder(BiNode<T>* bt)
{
if (bt == NULL) // 递归调用的结束条件
return;
cout << bt->data; // 访问根节点bt的数据域
PreOrder(bt->lchild); // 前序递归遍历bt的左子树
PreOrder(bt->rchild); // 前序递归遍历bt的右子树
} template<class T>
inline void BiTree<T>::InOrder(BiNode<T>* bt)
{
if (bt == NULL)
return;
InOrder(bt->lchild);
cout << bt->data;
InOrder(bt->rchild);
} template<class T>
inline void BiTree<T>::PostOrder(BiNode<T>* bt)
{
if (bt == NULL)
return;
PostOrder(bt->lchild);
PostOrder(bt->rchild);
cout << bt->data;
}
说明:1、除了层序遍历,其他遍历均为递归算法。
2、为什么层序遍历使用队列:在进行层序遍历时,对某一层的结点访问完后,再按照它们的访问次序对各个结点的左孩子和右孩子顺序访问,这样一层一层进行,先访问的结点其左右孩子也要先访问,这符合队列的操作特性,因此,在进行层序遍历时,可设置一个队列存放已访问的结点。
3、构造函数对二叉树的特殊处理:将二叉树中每个结点的空指针引出一个虚结点,其值为一特定值,如‘#’,以标识其为空。
4、二叉链表属于动态内存分配,需要在析构函数中释放二叉链表的所有结点。在释放某结点时,该结点的左右都子树已经释放,所以应该采用后序遍历。
3、主函数
#include"bitree.h"
using namespace std; int main()
{
BiTree<char>* bitree=new BiTree<char>(); // 创建一棵二叉树
bitree->PreOrder(); // 前序遍历
cout << endl;
bitree->InOrder(); // 中序遍历
cout << endl;
bitree->PostOrder(); // 后序遍历
cout << endl;
bitree->LeverOrder(); // 层序遍历
delete bitree; system("pause");
return ;
}
三、实例
建立如下二叉树,并输出四种遍历的结果。

运行结果:

结果正确。
参考文献:
[1]王红梅, 胡明, 王涛. 数据结构(C++版)[M]. 北京:清华大学出版社。
马上元旦了,祝大家元旦快乐!!2017-12-29
二叉树的二叉链表存储结构及C++实现的更多相关文章
- 建立二叉树的二叉链表存储结构(严6.70)--------西工大noj
#include <stdio.h> #include <stdlib.h> #include <string.h> typedef struct TreeNode ...
- C语言递归实现二叉树(二叉链表)的三种遍历和销毁操作(实验)
今天写的是二叉树操作的实验,这个实验有三个部分: ①建立二叉树,采用二叉链表结构 ②先序.中序.后续遍历二叉树,输出节点值 ③销毁二叉树 二叉树的节点结构定义 typedef struct BiTNo ...
- 二叉树(二叉链表实现)JAVA代码
publicclassTest{ publicstaticvoid main(String[] args){ char[] ch =newchar[]{'A','B ...
- 建立二叉树的二叉链表(严6.65)--------西工大noj
需要注意的点:在创建二叉树的函数中,如果len1==len2==0,一定要把(*T)置为NULL然后退出循环 #include <stdio.h> #include <stdlib. ...
- C#实现二叉树--二叉链表结构
二叉树的简单介绍 关于二叉树的介绍请看这里 : 二叉树的简单介绍 http://www.cnblogs.com/JiYF/p/7048785.html 二叉链表存储结构: 二叉树的链式存储结构是指,用 ...
- c使用二叉链表创建二叉树遇到的一些疑问和思考
二叉链表存储二叉树 学习的时候参考的是<大话数据结构>,书中是这样定义的 typedef char TElemType; typedef struct BiTNode { TElemTyp ...
- 【开200数组解决二叉搜索树的建立、遍历】PAT-L3-016. 二叉搜索树的结构——不用链表来搞定二叉搜索树
L3-016. 二叉搜索树的结构 二叉搜索树或者是一棵空树,或者是具有下列性质的二叉树: 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值:若它的右子树不空,则右子树上所有结点的值均大于它 ...
- PTA 7-2 二叉搜索树的结构(30 分)
7-2 二叉搜索树的结构(30 分) 二叉搜索树或者是一棵空树,或者是具有下列性质的二叉树: 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值:若它的右子树不空,则右子树上所有结点的值均大 ...
- 二叉搜索树的结构(30 分) PTA 模拟+字符串处理 二叉搜索树的节点插入和非递归遍历
二叉搜索树的结构(30 分) PTA 模拟+字符串处理 二叉搜索树的节点插入和非递归遍历 二叉搜索树的结构(30 分) 二叉搜索树或者是一棵空树,或者是具有下列性质的二叉树: 若它的左子树不空,则 ...
随机推荐
- Mac下配置maven和集成到ecclipse(Mac 10.12)
1.到官网下载maven,http://maven.apache.org/download.cgi 下载好的tar.gz包解压出来,并重命名为maven3,拷贝到/usr/local目录下 2.配置环 ...
- mysql故障总结
MYSQL故障排查 https://zhuanlan.zhihu.com/p/27834293
- ruby中的retry和redo
# retry用于处理异常中的begin/end代码块中,如果发生异常会重新运行 begin 和 rescue 之间的代码#当retry 被调用的话,begin 和 rescue 之间所有的代码都会被 ...
- 【Docker】Dockerfile使用apt-get来安装jdk
前面谈过使用wget来从oracle下载jdk安装文件是使用了cookie欺骗的方法来越过身份验证来使用Dockerfile在ubuntu内安装oracle版本的jdk的. 然而正道还是用apt-ge ...
- C 标准库 - string.h之strpbrk使用
strpbrk Locate characters in string,Returns a pointer to the first occurrence in str1 of any of the ...
- Java入门系列-16-继承
这一篇文章教给新手学会使用继承,及理解继承的概念.掌握访问修饰符.掌握 final 关键字的用法. 继承 为什么要使用继承 首先我们先看一下这两个类: public class Teacher { p ...
- Golang教程:goroutine协程
在上一篇中,我们讨论了并发,以及并发和并行的区别.在这篇教程中我们将讨论在Go中如何通过Go协程实现并发. 什么是协程 Go协程(Goroutine)是与其他函数或方法同时运行的函数或方法.可以认为G ...
- MVVM - 事件转命令2
在使用MVVM模式时, 按照模式的规则是尽量不直接使用事件. 所以对于以前一直使用事件模式的同行来说确实有点头疼. 还好微软给我们提供了几种间接使用事件(命令)的方法, 下面我就来看看这几种方法: I ...
- JRebel - 给IDE安装JRebel插件
JRebel对于很多人来说已经并不陌生了,一搜一大把. 用过JRebel后发现,这对于Java开发简直不可缺少. 尽管其价格有点春节国庆期间的各种交通费用——打劫! 即使如此也出现了有"分享 ...
- mybatis学习之CLOB、BLOB处理及多参数方法映射
CLOB数据mysql对应数据类型为longtext.BLOB类型为longblob: model实体: ... private Integer id; private String name; pr ...