二叉查找树C++实现
二分查找树特点:
(1) 若任意节点的左子树不空,则左子树上所有结点的值均小于它的根结点的值;
(2) 任意节点的右子树不空,则右子树上所有结点的值均大于它的根结点的值;
(3) 任意节点的左、右子树也分别为二叉查找树。
(4) 没有键值相等的节点(no duplicate nodes)。
前序遍历:中左右
中序遍历:左中右
序遍历:左右中
二叉查找树的重点在于如何找节点的前驱节点和后继节点
#pragma once
#include <iostream>
using namespace std; template <class T>
class BSTNode
{
public:
T key;
BSTNode *parent;
BSTNode *left;
BSTNode *right; BSTNode(T value, BSTNode *p, BSTNode *l, BSTNode *r):key(value),parent(p),left(l),right(r)
{ }
}; template <class T>
class BSTree
{
private:
BSTNode<T> *mRoot; public:
BSTree():mRoot(NULL){}
~BSTree(){} // 前序排序
void preOrder()
{
preOrder(mRoot);
}
void inOrder()
{
inOrder(mRoot);
}
void postOrder()
{
postOrder(mRoot);
}
// 查找二叉树中键值为key的节点
BSTNode<T>* SearchKey(const T key)
{
return SearchKey(mRoot, key);
}
BSTNode<T>* minKey()
{
return minKey(mRoot);
}
BSTNode<T>* maxKey()
{
return maxKey(mRoot);
}
// 插入节点
void insert( T key)
{
BSTNode<T> *z = new BSTNode<T>(key, NULL, NULL, NULL); if (z == NULL)
{
return;
}
insert(mRoot, z);
} private:
// 前序排序
void preOrder(BSTNode<T> *tree) const
{
if (tree != NULL)
{
cout << tree->key << " ";
preOrder(tree->left);
preOrder(tree->right);
}
} // 中序排序
void inOrder(BSTNode<T> *tree) const
{
if (tree != NULL)
{
preOrder(tree->left);
cout << tree->key << " ";
preOrder(tree->right);
}
} // 后序排序
void postOrder(BSTNode<T> *tree) const
{
if (tree != NULL)
{
preOrder(tree->left);
preOrder(tree->right);
cout << tree->key << " ";
}
}
BSTNode<T>* SearchKey(BSTNode<T>* pNode, const T key) const
{
// 递归查找
/*if (pNode = NULL || key == pNode->key)
{
return pNode;
}
else if (key > pNode->key)
{
return SearchKey(pNode->right);
}
else
{
return SearchKey(pNode->left);
}*/ // 非递归查找
BSTNode<T>* x = pNode;
while (x != NULL)
{
if (key > x->key)
{
x = x->right;
}
else if (key < x->key)
{
x = x->left;
}
else
{
return x;
}
} return NULL;
}
// 将节点插入到二叉树中
void insert(BSTNode<T>* &tree, BSTNode<T> *Node)
{
BSTNode<T> *y = NULL;
BSTNode<T> *x = tree;
while (NULL != x)
{
y = x;
if (Node->key > x->key)
{
x = x->right;
}
else
{
x = x->left;
}
} Node->parent = y; // 这到后面两句为关键代码
if (NULL == y)
{
tree = Node;
}
else if (Node->key > y->key)
{
y->right = Node;
}
else
{
y->left = Node;
}
}
// 查找最小节点
BSTNode<T>* minKey(BSTNode<T>* pNode) const
{
while (pNode != NULL)
{
pNode = pNode->left;
} return pNode;
}
BSTNode<T>* maxKey(BSTNode<T>* pNode) const
{
while (pNode != NULL)
{
pNode = pNode->right;
} return pNode;
}
// 找节点(x)的后继节点。即查找二叉树中数值大于该节点的最小值
BSTNode<T>* Successor(BSTNode<T>* x)
{
// 分三种情况
// 1. x有右孩子,找到以右孩子为根的子树的最小节点
// 2. x没有右孩子,当x为左孩子,则x的父节点为后继节点
// 2. x没有右孩子,当x为右孩子,则找x的最低父节点,并且该父节点具有左孩子
if (x->right != NULL)
{
return minKey(x->right);
}
BSTNode<T>* y = x->parent;
while ((NULL != y) &&(x == y->right))
{
x= y;
y = y->parent;
} return y;
}
// 找结点(x)的前驱结点。即查找"二叉树中数据值小于该结点"的"最大结点"
BSTNode<T>* BSTree<T>::predecessor(BSTNode<T> *x)
{
// 如果x存在左孩子,则"x的前驱结点"为 "以其左孩子为根的子树的最大结点"。
if (x->left != NULL)
return maxKey(x->left); // 如果x没有左孩子。则x有以下两种可能:
// (01) x是"一个右孩子",则"x的前驱结点"为 "它的父结点"。
// (01) x是"一个左孩子",则查找"x的最低的父结点,并且该父结点要具有右孩子",找到的这个"最低的父结点"就是"x的前驱结点"。
BSTNode<T>* y = x->parent;
while ((y!=NULL) && (x==y->left))
{
x = y;
y = y->parent;
} return y;
} // 删除二叉树中的节点,并返回被删除的节点
//BSTNode<T>* RemoveNode(BSTNode<T>* &tree, BSTNode<T>* pNode)
//{
// BSTNode<T>* x = tree; // while (NULL != x && pNode->key != x->key)
// {
// if (pNode->key > x->key)
// {
// x = x->right;
// }
// else if (pNode->key < x->key)
// {
// x = x->left;
// }
// } // // 找到或x为空 //}
};
二叉查找树C++实现的更多相关文章
- 数据结构:二叉查找树(C语言实现)
数据结构:二叉查找树(C语言实现) ►写在前面 关于二叉树的基础知识,请看我的一篇博客:二叉树的链式存储 说明: 二叉排序树或者是一棵空树,或者是具有下列性质的二叉树: 1.若其左子树不空,则左子树上 ...
- 数据结构笔记--二叉查找树概述以及java代码实现
一些概念: 二叉查找树的重要性质:对于树中的每一个节点X,它的左子树任一节点的值均小于X,右子树上任意节点的值均大于X. 二叉查找树是java的TreeSet和TreeMap类实现的基础. 由于树的递 ...
- codevs 1285 二叉查找树STL基本用法
C++STL库的set就是一个二叉查找树,并且支持结构体. 在写结构体式的二叉查找树时,需要在结构体里面定义操作符 < ,因为需要比较. set经常会用到迭代器,这里说明一下迭代器:可以类似的把 ...
- 平衡二叉查找树(AVL)的理解与实现
AVL树的介绍 平衡二叉树,又称AVL(Adelson-Velskii和Landis)树,是带有平衡条件的二叉查找树.这个平衡条件必须要容易保持,而且它必须保证树的深度是 O(log N).一棵AVL ...
- 二叉查找树 C++实现(含完整代码)
一般二叉树的查找是通过遍历整棵二叉树实现,效率较低.二叉查找树是一种特殊的二叉树,可以提高查找的效率.二叉查找树又称为二叉排序树或二叉搜索树. 二叉查找树的定义 二叉排序树(Binary Search ...
- 数据结构——二叉查找树、AVL树
二叉查找树:由于二叉查找树建树的过程即为插入的过程,所以其中序遍历一定为升序排列! 插入:直接插入,插入后一定为根节点 查找:直接查找 删除:叶子节点直接删除,有一个孩子的节点删除后将孩子节点接入到父 ...
- Java for LintCode 验证二叉查找树
给定一个二叉树,判断它是否是合法的二叉查找树(BST) 一棵BST定义为: 节点的左子树中的值要严格小于该节点的值. 节点的右子树中的值要严格大于该节点的值. 左右子树也必须是二叉查找树. ...
- 数据结构和算法 – 9.二叉树和二叉查找树
9.1.树的定义 9.2.二叉树 人们把每个节点最多拥有不超过两个子节点的树定义为二叉树.由于限制子节点的数量为 2,人们可以为插入数据.删除数据.以及在二叉树中查找数据编写有效的程序了. 在 ...
- 二叉树-二叉查找树-AVL树-遍历
一.二叉树 定义:每个节点都不能有多于两个的儿子的树. 二叉树节点声明: struct treeNode { elementType element; treeNode * left; treeNod ...
- 二叉查找树的Java实现
为了克服对树结构编程的恐惧感,决心自己实现一遍二叉查找树,以便掌握关于树结构编程的一些技巧和方法.以下是基本思路: [1] 关于容器与封装.封装,是一种非常重要的系统设计思想:无论是面向过程的函数,还 ...
随机推荐
- mycat安装与配置
1.安装jdk 测试jdk是否已经安装 [root@node002 ~]# java -version-bash: java: command not found 创建解压目录 [root@node0 ...
- Java框架之Spring MVC(一)
一.Spring简介 Spring MVC是当前最优秀的 MVC 框架,自从Spring 2.5 版本发布后,由于支持注解配置,易用性有了大幅度的提高.Spring 3.0 更加完善,实现了对 Str ...
- C语言中处理结构体的原理
汇编中有几种寻址方式,分别是直接寻址:(ds:[idata]).寄存器间接寻址(ds:[bx]).寄存器相对寻址(ds:[bx + idata].ds:[bx + si])基址变址寻址(ds:[bx ...
- kylin客户端(python编写)不能按照预期的segment进行rebuild
kylin_client_tool 提供了对cube进行BUILD,REBUILD,MERGE功能,其中REBUILD却不能达到预期的效果按照指定的segment执行. 场景: 当我在kylin we ...
- 【转】qlv文件如何转换成mp4 怎样把下载好的qlv格式视频转换成MP4格式
狸窝 复制 收藏 保存到桌面 快速找教程方案 反馈需求 社会主义核心价值观 客服QQ41442901 马上注册 升级VIP 对于视频文件之间的转换问题,我也已经是无力吐槽了,每个 ...
- 开blog
开这个blog,现阶段还是主要作为自己的学习笔记 If it could help others, it would be better!
- OOAD-设计模式(一)概述
前言 在我们很多时候设计代码都是需要用到各种不一样的设计模式的,接下来着几篇给大家领略一下设计模式.知道设计模式的作用,以及在代码的具体体现.很多时候我们看不懂代码就是因为我们不知道它使用的设计模式. ...
- linux(二)之linux常用命令一
linux的命令非常之多,命令多就算了关键每个命令还有很多的参数.不过其实并不要去害怕它.你只要常去用,并且的话,大部分你只要记住命令,参数不记得我们可以去查帮助文档.加油吧!老帖们. 一.登入\登出 ...
- 三 : spring-uploadify上传文件
一 : applicationContext.xml中:必须声明不然获取不到<!-- 上传文件的配置 --> <bean id="multipartResolver&quo ...
- 废旧鼠标先别丢,用来学习nRF52832 的QDEC
刚发现nRF52832有一个 QDEC,SDK13.0中还有驱动,但是不太友好. 如果大家有废旧鼠标,建议拆一个编码器下来“学习”.鼠标的一般原理如下: 图一 图中那个SW4 ALPS EC10E ...