数据结构之二叉树篇卷四 -- 二叉树线索化(With Java)
一、线索二叉树简介
二叉树本身是一种非线性结构,然而当你对二叉树进行遍历时,你会发现遍历结果是一个线性序列。这个序列中的节点存在前驱后继关系。因此,如何将这种前驱后继信息赋予给原本的二叉树呢?这就是二叉树的线索化过程。所以可以通过丰富原有的二叉树构建一棵可以知道结点的前驱后继的新的二叉树,我们叫它线索二叉树。
二、构建线索二叉树
2.1 定义线索二叉树节点
public class ThreadNode<E> { public E data;
public ThreadNode<E> lnode;
public ThreadNode<E> rnode; enum tag {CHILD, THREAD};
public tag ltag;
public tag rtag; public ThreadNode(E data){
this.data = data;
ltag = tag.CHILD;
rtag = tag.CHILD;
} public ThreadNode(){}
}
为了充分利用二叉树的空指针域,我们可以将结点线索依附在这些空指针中。这样一来,在我们访问根节点的左右节点时,我们必须要区分什么时候是线索,什么时候子女指针。因此为每个节点设置两个标志位。
2.2 建立线索二叉树
在遍历二叉树的时候顺便进行线索化。线索化就是每遇到一个节点存在空指针域就要立即填上它的前驱(左指针空)或者后继(右指针空)。
/**
* 利用中序非递归遍历对二叉树进行线索化
* @author sheepcore
*/
public void createInThread(){
if(root == null)
return;
ThreadNode<E> pre = null;
ThreadNode<E> cur = root;
MyStack<ThreadNode<E>> stack = new MyStack<>();
while (cur != null || !stack.isEmpty()) {
while (cur != null){
stack.push(cur);
cur = cur.lnode;
}
if(!stack.isEmpty()){
cur = stack.pop();
if(cur.lnode == null) { //当前节点的前驱作为线索
cur.lnode = pre;
cur.ltag = ThreadNode.tag.THREAD;
}
if(pre != null && pre.rnode == null){
pre.rnode = cur; //当前节点的前驱的后继节点是当前节点
pre.rtag = ThreadNode.tag.THREAD;
}
pre = cur;
if(cur.rnode != null){
cur = cur.rnode;
}
else
cur = null; } //if
} //while
}
2.3 遍历线索二叉树
以中序线索二叉树的遍历为例。getFirst(root) 和 getNext(root) 分别是获取中序遍历中以 root 为根的第一个访问的节点和 root 的后继节点。这两个辅助函数在最后面的完整代码有实现。
public void inOrder(ThreadNode<E> root){
ThreadNode<E> cur;
for (cur = getFirst(root); cur != null; cur = getNext(cur)) {
visit(cur);
}
}
2.4 完整代码
public class ThreadTree <E> { private ThreadNode<E> root; /**
* using for construct a binary tree with its element as an integer
*/
public ArrayList<Integer> list; public ThreadNode<E> getRoot() { return root; } public ThreadTree(){
list = new ArrayList<>();
} /**
* build a binary tree in a preorder way recursively
* @return root
* @author sheepcore
*/
public ThreadNode<E> generate() {
if(list.isEmpty())
return null;
int data = list.get(0);
list.remove(0);
ThreadNode<E> node; if(data != -1) {
node = (ThreadNode<E>) new ThreadNode<Integer>(data);
node.lnode = generate(); //generate left subtree
node.rnode = generate(); //generate right subtree
return node;
}
else
return null;
} /**
* build a binary tree in preorder
* @param treeSequence "1 2 -1 -1 3 -1 -1 "
*/
public void preOderBuild(String treeSequence) {
String[] nodes = treeSequence.split("\\s+");
for (int i = 0; i < nodes.length; i++) {
list.add(Integer.parseInt(nodes[i]));
}
root = generate();
} /**
* 利用中序非递归遍历对二叉树进行线索化
* @author sheepcore
*/
public void createInThread(){
if(root == null)
return;
ThreadNode<E> pre = null;
ThreadNode<E> cur = root;
MyStack<ThreadNode<E>> stack = new MyStack<>();
while (cur != null || !stack.isEmpty()) {
while (cur != null){
stack.push(cur);
cur = cur.lnode;
}
if(!stack.isEmpty()){
cur = stack.pop();
if(cur.lnode == null) { //当前节点的前驱作为线索
cur.lnode = pre;
cur.ltag = ThreadNode.tag.THREAD;
}
if(pre != null && pre.rnode == null){
pre.rnode = cur; //当前节点的前驱的后继节点是当前节点
pre.rtag = ThreadNode.tag.THREAD;
}
pre = cur;
if(cur.rnode != null){
cur = cur.rnode;
}
else
cur = null; } //if
} //while
} public ThreadNode<E> getFirst(ThreadNode<E> root){
ThreadNode<E> cur = root;
while (cur != null && cur.ltag == ThreadNode.tag.CHILD){
cur = cur.lnode;
}
return cur;
} public ThreadNode<E> getLast(ThreadNode<E> root){
ThreadNode<E> cur = root;
while (cur != null && cur.rtag == ThreadNode.tag.CHILD){
cur = cur.rnode;
}
return cur;
} public ThreadNode<E> getNext(ThreadNode<E> root){
ThreadNode<E> cur = root;
if(cur.rtag == ThreadNode.tag.THREAD)
return cur.rnode;
else
return getFirst(cur.rnode); } public ThreadNode<E> getPrior(ThreadNode<E> root){
ThreadNode<E> cur = root;
if(cur.ltag == ThreadNode.tag.THREAD)
return cur.lnode;
else
return getLast(cur.lnode); } public void visit(ThreadNode<E> root){
if(root != null && root.data != null){
System.out.print(root.data);
System.out.print("\t");
}
} /**
* inorder traversal
* @param root
* @author sheepcore
*/
public void inOrder(ThreadNode<E> root){
ThreadNode<E> cur;
for (cur = getFirst(root); cur != null; cur = getNext(cur)) {
visit(cur);
}
} public static void main(String[] args) { String str = "0 1 -1 3 -1 -1 2 4 -1 -1 -1";
ThreadTree<Integer> threadTree = new ThreadTree<>();
threadTree.preOderBuild(str);
threadTree.createInThread();
threadTree.inOrder(threadTree.getRoot());
}
}
数据结构之二叉树篇卷四 -- 二叉树线索化(With Java)的更多相关文章
- 算法与数据结构(三) 二叉树的遍历及其线索化(Swift版)
前面两篇博客介绍了线性表的顺序存储与链式存储以及对应的操作,并且还聊了栈与队列的相关内容.本篇博客我们就继续聊数据结构的相关东西,并且所涉及的相关Demo依然使用面向对象语言Swift来表示.本篇博客 ...
- 数据结构之二叉树篇卷三 -- 二叉树非递归遍历(With Java)
Nonrecursive Traversal of Binary Tree First I wanna talk about why we should <code>Stack</c ...
- 数据结构之二叉树篇卷二 -- 二叉树递归遍历(With Java)
一.先序递归遍历(Preorder Recursive Traversal) 1.1 算法 首先需要明确的是这里的序是针对 root 节点而言的.故先序即先“访问”根节点,其次“访问”其左右节点. 1 ...
- 【数据结构&算法】12-线索二叉树
目录 前言 线索二叉树的概念 线索二叉树的实现 线索二叉树的寻点思路二 类双向链表参考图 参考代码 中序遍历线索化 前言 在<大话数据结构>P190 页中有一句话:其实线索二叉树,就等于是 ...
- 遍历二叉树 traversing binary tree 线索二叉树 threaded binary tree 线索链表 线索化
遍历二叉树 traversing binary tree 线索二叉树 threaded binary tree 线索链表 线索化 1. 二叉树3个基本单元组成:根节点.左子树.右子树 以L.D.R ...
- 【java 数据结构】还不会二叉树?一篇搞定二叉树
二叉树是我们常见的数据结构之一,在学习二叉树之前我们需要知道什么是树,什么是二叉树,本篇主要讲述了二叉树,以及二叉树的遍历. 你能get到的知识点? 1.树的介绍 2.二叉树的介绍 3.二叉树遍历的四 ...
- c++(排序二叉树线索化)
前面我们谈到了排序二叉树,还没有熟悉的同学可以看一下这个,二叉树基本操作.二叉树插入.二叉树删除1.删除2.删除3.但是排序二叉树也不是没有缺点,比如说,如果我们想在排序二叉树中删除一段数据的节点怎么 ...
- JS数据结构第五篇 --- 二叉树和二叉查找树
一.二叉树的基本概念 从逻辑结构角度来看,前面说的链表.栈.队列都是线性结构:而今天要了解的“二叉树”属于树形结构. 1.1 多叉树的基本概念,以上图中“多叉树”为例说明 节点:多叉树中的每一个点都叫 ...
- JS数据结构第六篇 --- 二叉树力扣练习题
1.第226题:翻转二叉树 递归+迭代两种实现方式: /** 反转二叉树 * Definition for a binary tree node. * function TreeNode(val) { ...
随机推荐
- BZOJ 1923: [Sdoi2010]外星千足虫 高斯消元+bitset
高斯消元求解异或方程组,可以多学一下 $bitset$ 在位运算中的各种神奇操作. #include <cstdio> #include <bitset> #define N ...
- ZOJ - 4045District Division dfs划分子树
ZOJ - 4045District Division 题目大意:给你n个节点的树,然后让你划分这棵数使得,每一块都恰好k个节点并且两两间是连通的,也就是划分成n/k个连通集,如果可以输出YES,并输 ...
- java中如何补齐汉字字符
一个汉字相当于两个字符,所以需要输入法的时候切换到[中文全角],中文全角占用2个字符(一个空格),半角占用1个字符
- [笔记]动态规划(dynamic programming)
动态规划与分治方法都是通过组合子问题的解来求解原问题,区别在于:分治方法将问题划分为互不相交的子问题,递归求解子问题,再将它们的解组合起来,求出原问题的解.分治算法可能反复的求解某些公共子问题,从而使 ...
- PHPstrom中关闭提示信息
设置里面搜索parameter name hints
- JavaScript 正则的使用方法
JavaScript正则方法 1.compile 编译一个正则表达式对象 rgExp.compile(pattern, [flags]) pattern字符串表达式 2.exec 在指定字符串中执行 ...
- FPGA实战操作(2) -- PCIe总线(例程设计分析)
1.框架总览 平台:vivado 2016.4 FPGA:A7 在实际应用中,我们几乎不可能自己去编写接口协议,所以在IP核的例程上进行修改来适用于项目是个不错的选择. 通过vivado 中有关PCI ...
- Mac 下 Android Studio 连 夜神模拟器 调试
1.运行夜神模拟器(自行搜索下载) 2.进入资源目录 cd /Applications/Nox\ App\ Player.app/Contents/MacOS 3.执行命令连接 adb connect ...
- 爬虫 lxml 模块
Xpath 在 XML 文档中查找信息的语言, 同样适用于 HTML 辅助工具 Xpath Helper Chrome插件 快捷键 Ctrl + shift + x XML Quire xpath ...
- UDP打洞原理介绍
NAT穿越模块的设计与实现 Internet的快速发展以及IPv4地址数量的不足使得NAT设备得到了大规模的应用,然而这也给越来越多的端到端通信也带来了不少的麻烦.一般来说,NAT设备允许内网内主机 ...