#include <iostream>

using namespace std;

//AVL树的节点
template<typename T>
class TreeNode
{
public:
TreeNode() :lson(NULL), rson(NULL), freq(), hgt(){}
T data;//值
int hgt;//以这个结点为根的树的高度
int freq;//相同点的频率,我是不知道
TreeNode* lson, *rson;//左右儿子的地址
}; template<typename T>
class AVLTree
{//AVLTree的类属性和方法声明
private:
TreeNode<T>*root;//根节点
void insertpri(TreeNode<T>*&node, T x);//插入
TreeNode<T>* findpri(TreeNode<T>* node, T x);//查找
void traversalpri(TreeNode<T>* node);//遍历
void delpri(TreeNode<T>* &node, T x);//删除
int height(TreeNode<T>* node);//辅助操作:高度
void singLeft(TreeNode<T>* &k2);//左左操作
void singRight(TreeNode<T>* &k2);//右右操作
void doubleLeft(TreeNode<T>* &k3);//左右操作
void doubleRight(TreeNode<T>* &k3);//右左操作
int max(int cmpa, int cmpb);
public:
AVLTree() :root(NULL){};
void insert(T x);//插入接口
void del(T x);//删除接口
TreeNode<T>* find(T x);//查找接口
void traversal();//遍历接口
}; //辅助操作:高度
template<typename T>
int AVLTree<T>::height(TreeNode<T>* node)
{
if (node!=NULL)
return node->hgt;
return -;
} //辅助操作:比较高度
template<typename T>
int AVLTree<T>::max(int cmpa, int cmpb)
{
return cmpa > cmpb ? cmpa : cmpb;
} template<typename T>
void AVLTree<T>::singLeft(TreeNode<T>* &k2)
{//左左情况下旋转
TreeNode<T>* k1;
k1 = k2->lson;
k2->lson = k1->rson;
k1->rson = k2;
k2 = k1;
k2->hgt = max(height(k2->lson), height(k2->rson))+;
k1->hgt = max(height(k1->lson), height(k1->rson))+;
} template<typename T>
void AVLTree<T>::singRight(TreeNode<T>* &k2)
{//左左情况下旋转
TreeNode<T>* k1;
k1 = k2->rson;
k2->rson = k1->lson;
k1->lson = k2;
k2 = k1;
k2->hgt = max(height(k2->lson), height(k2->rson))+;
k1->hgt = max(height(k1->lson), height(k1->rson))+;
} template<typename T>
void AVLTree<T>::doubleLeft(TreeNode<T>* &k3)
{//左右的情况
singRight(k3->lson);
singLeft(k3);
} //右左的情况
template<typename T>
void AVLTree<T>::doubleRight(TreeNode<T>* &k3)
{
singLeft(k3->rson);
singRight(k3);
} //插入
template<typename T>
void AVLTree<T>::insertpri(TreeNode<T>* &node, T x)
{
if (node == NULL)
{
node = new TreeNode<T>();
node->data = x;
return;
}
if (node->data>x)
{
insertpri(node->lson, x);//递归插入
//插入后自我调整
if ( == height(node->lson) - height(node->rson))
if (node->lson->data < x)
doubleRight(node);
else
singLeft(node);
}
else if (node->data < x)
{
insertpri(node->rson, x);
if ( == height(node->rson) - height(node->lson))
if (node->rson->data < x)
singRight(node);
else
doubleRight(node);
}
else ++(node->freq);
//取新的高度值,后面的+1很重要,作者都忘记了加
node->hgt = max(height(node->lson), height(node->rson)) + ;
} //插入接口
template<typename T>
void AVLTree<T>::insert(T x)
{
insertpri(root, x);
} //查找
template<typename T>
TreeNode<T>* AVLTree<T>::findpri(TreeNode<T>* node, T x)
{
if (node == NULL)
return NULL;
if (node->data > x)
return findpri(node->lson, x);
else if (node->data < x)
return findpri(node->rson, x);
else return node;
} //查找
template<typename T>
TreeNode<T>* AVLTree<T>::find(T x)
{
findpri(root, x);
} //删除
template<typename T>
void AVLTree<T>::delpri(TreeNode<T>* &node,T x)
{
if (node == NULL)
return;
if (x < node->data)
{
delpri(node->lson, x);
//删除后的调整
if ( == height(node->rson) - height(node->lson))
if (node->rson->lson&&height(node->rson->lson) > height(node->rson->rson))
doubleRight(node);
else
singRight(node);
}
else if (x > node->data)
{
delpri(node->rson, x);
if ( == height(node->lson) - height(node->rson))
if (node->lson->rson&&height(node->lson->rson) > height(node->lson->lson))
doubleLeft(node);
else
singLeft(node);
}
else//找到后的操作
{//先是有两个儿子的情况
if (node->lson&&node->rson)
{
TreeNode<T>* t = node->lson;
for (; t->rson; t = t->rson);
node->data = t->data;
node->freq = t->freq;
//递归到一个儿子或没有儿子的情况
delpri(node->lson, t->data);
if ( == height(node->rson) - height(node->lson))
{//下面的if自己不会写
if (node->rson->lson&&height(node->rson->lson) > height(node->rson->rson))
doubleRight(node);
else
singRight(node);
}
}
else
{
TreeNode<T>* t = node;
if (node->lson == NULL)
node = node->rson;
else if (node->rson == NULL)
node = node->lson;
delete(t);
t = NULL;
}
}
if (node == NULL)return;//表示只有根节点,删了之后就没有了
node->hgt = max(height(node->lson), height(node->rson));
return;
} template<typename T>
void AVLTree<T>::del(T x)
{
delpri(root, x);
} //中序遍历函数
template<class T>
void AVLTree<T>::traversalpri(TreeNode<T>* node)
{
if (node == NULL) return;
traversalpri(node->lson);//先遍历左子树
cout << node->data << " ";//输出根节点
traversalpri(node->rson);//再遍历右子树
}
//中序遍历接口
template<class T>
void AVLTree<T>::traversal()
{
traversalpri(root);
} int main()
{
AVLTree<int> t;
t.insert();
t.insert();
t.insert();
t.insert();
t.insert();
t.insert();
t.del();
t.insert();
t.traversal();
}

AVL树相关操作的更多相关文章

  1. AVL树插入操作实现

    为了提高二插排序树的性能,规定树中的每个节点的左子树和右子树高度差的绝对值不能大于1.为了满足上面的要求需要在插入完成后对树进行调整.下面介绍各个调整方式. 右单旋转 如下图所示,节点A的平衡因子(左 ...

  2. avl树的操作证明

    以下用大O表示节点,ABC表示三个集合. 仅分析左子树的情况,因为对称,右子树的情况一样. 插入节点前 O /     \ O        A   /    \ B       C 插入节点后: O ...

  3. 数据结构树之AVL树(平衡二叉树)

    一 什么是AVL树(平衡二叉树): AVL树本质上是一颗二叉查找树,但是它又具有以下特点:它是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树.在AVL树中任何节 ...

  4. 图解数据结构树之AVL树

    AVL树(平衡二叉树): AVL树本质上是一颗二叉查找树,但是它又具有以下特点:它是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树.在AVL树中任何节点的两个子 ...

  5. AVL树Python实现

    # coding=utf-8 # AVL树Python实现 def get_height(node): return node.height if node else -1 def tree_mini ...

  6. 树-二叉搜索树-AVL树

    树-二叉搜索树-AVL树 树 树的基本概念 节点的度:节点的儿子数 树的度:Max{节点的度} 节点的高度:节点到各叶节点的最大路径长度 树的高度:根节点的高度 节点的深度(层数):根节点到该节点的路 ...

  7. AVL树(平衡二叉树)

    定义及性质 AVL树:AVL树是一颗自平衡的二叉搜索树. AVL树具有以下性质: 根的左右子树的高度只差的绝对值不能超过1 根的左右子树都是 平衡二叉树(AVL树) 百度百科: 平衡二叉搜索树(Sel ...

  8. AVL树(查找、插入、删除)——C语言

    AVL树 平衡二叉查找树(Self-balancing binary search tree)又被称为AVL树(AVL树是根据它的发明者G. M. Adelson-Velskii和E. M. Land ...

  9. 深入理解索引和AVL树、B-树、B+树的关系

    目录 什么是索引 索引的分类 索引和AVL树.B-树.B+树的关系 AVL树.红黑树 B-树 B+树 SQL和NoSQL索引 什么是索引 索引时数据库的一种数据结构,数据库与索引的关系可以看作书籍和目 ...

随机推荐

  1. 如何使用sublime编辑器运行python程序

    现在越发喜欢sublime编辑器了,不仅界面友好美观.文艺,可扩展性还特别强. sublime本身是不具备运行python程序的能力的,需要做些设置才可以.以下是安装好sublime后设置的步骤: 点 ...

  2. C程序设计语言练习题1-20

    练习1-20 编写程序detab,将输入中的制表符替换成适当数目的空格,使空格充满到下一个制表符终止位的地方.假设制表符终止位的位置是固定的,比如每隔n列就会出现一个制表符终止位.n应该是变量还是符号 ...

  3. 大整数算法[11] Karatsuba乘法

    ★ 引子         前面两篇介绍了 Comba 乘法,最后提到当输入的规模很大时,所需的计算时间会急剧增长,因为 Comba 乘法的时间复杂度仍然是 O(n^2).想要打破乘法中 O(n^2) ...

  4. web登录与授权

    web开发已经流行了很多年,登录与授权也基本有一套通用的流程,下面是我自己常用的登录与授权方式,欢迎大家讨论与吐槽. 概念: 登录是过程,授权是结果.登录只是为了获得页面的访问权限 or 操作权限 o ...

  5. Solr自动生成ID

    在Solr中,每一个索引,都要有一个唯一的ID,类似于关系型数据库表中的主键.为了方便创建索引,需要配置自动生成的ID,即UUID. 一.配置schema.xml文件 添加uuid字段类型,修改字段i ...

  6. Swift—类的继承-备

    Swift中的继承只能发生在类上,不能发生在枚举和结构体上.一个类可以继承另一个类的方法.属性.下标等特征,当一个类继承其他类时,继承类叫子类,被继承类叫父类(或超类).子类继承父类后,可以重写父类的 ...

  7. Xcode 如何设置 自动内存管理 转换为 手动内存管理

    建议使用自动内存管理 ARC. 如果不想自动内存管理,可以在build phases 下的compile sources 中找到不想自动管理的.m文件 ,给它加compiler flags 为 -fn ...

  8. 转:fopen与open可以转换吗

    绝对不可以.fopen是C运行库级别的函数,而open是system call的wrapper routine.fopen返回FILE *的指针,这个结构本身维护着一些关于这个文件的信息,而open返 ...

  9. Uubntu E: Sub-process /usr/bin/dpkg returned an error code问题的解决办法

      cd /var/lib/dpkg   sudo mv info info.bak   sudo mkdir info      apt-get --reinstall install udev m ...

  10. android ftp案例分析

    使用方法: FTPClient client = new FTPClient(); client.connect("ftp.host.com", 8021); client.log ...