二叉查找树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] 关于容器与封装.封装,是一种非常重要的系统设计思想:无论是面向过程的函数,还 ...
随机推荐
- vue-父子组件嵌套的示例
组件注册: // 注册组件 Vue.component('my-component', { template: '<div>A custom component!</div>' ...
- mimtproxy和arpspoof实现局域网MITM
本地环境 环境:kali系统 目标机器:192.168.0.101 局域网网关:192.168.0.1 当前网络网卡端口:wlan0 arp欺骗流程 命令行开启本地数据转发: echo > /p ...
- for循环找出2到100的质数(素数)
思路: 1,一个数只有1和它本身两个因数,这个数叫质数. 2.注意:缩进这里else是for循环这个上下文的. 代码: for num in range(2,100): #为大循环变量num提供2-1 ...
- js中的for循环案例
打印99乘法表 for(var x=1; x<=9; x++) { for(var y=1; y<=x; y++) { document.write(y+"*&q ...
- Video Target Tracking Based on Online Learning—TLD单目标跟踪算法详解
视频目标跟踪问题分析 视频跟踪技术的主要目的是从复杂多变的的背景环境中准确提取相关的目标特征,准确地识别出跟踪目标,并且对目标的位置和姿态等信息精确地定位,为后续目标物体行为分析提供足 ...
- Offcanvas 自适应窗口示例
<!DOCTYPE html><html lang="zh-CN"><head> <meta charset="UTF-8&qu ...
- js设计模式之惰性单例模式
<html> <body> <button id="loginBtn">登录</button> </body> < ...
- python写一个翻译的小脚本
起因: 想着上学看不懂English的PDF感慨万分........ 然后就有了翻译的脚本. 截图: 代码: #-*- coding:'utf-8' -*- import requests impor ...
- SpringBoot(一)走进Springboot的世界
什么是spring boot Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程.该框架使用了特定的方式来进行配置,从而使开发人员 ...
- HDU6237-A Simple Stone Game-找素因子(欧拉函数)-2017中国大学生程序设计竞赛-哈尔滨站-重现赛
A Simple Stone Game Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Ot ...