AVL学习笔记

AVL,平衡二叉查找树。删除,插入,查找的复杂度都是O(logn)。它是一棵二叉树。对于每个节点来说,它的左孩子的键值都小于它,右孩子的键值都大于它。对于任意一个节点,它的左右孩子的高度差不大于1。树的高度的定义为:空节点的高度为0,非空节点的高度为左右孩子高度的最大值加1。

在插入删除过程中,会出现不平衡的时候。这时,会通过以下方式进行旋转保持树的平衡。下图中每一列最后一行是旋转后的结果,上面两行是对应的初始化状态。

1 插入。在以某个节点为根的子树中插入一个节点后,有可能使得该节点的左右子树的高度差大于1(其实此时的高度差是2),那么视情况进行LL,RR,LR,RL四种旋转中的一种可维持树的平衡。

2 删除。删除的键值小于当前节点键值时,在左子树中删除;大于当前节点键值时在右子树中进行删除;否则就是删除当前节点。删除当前节点时,找到后继节点,然后将后继结点替换当前节点,然后递归地删除这个后继结点即可。

template<class _ValyeType,class _FuncType>
class CAVLTree
{
protected:
    struct AVLTreeNode
    {
        _ValyeType m_iValue;
        AVLTreeNode* m_pLeftSon;
        AVLTreeNode* m_pRightSon;
        int m_nHeight;
        int m_nValueNumber;
    };
    AVLTreeNode* m_pRoot;
    _FuncType* m_pCompareFunc;

    AVLTreeNode* _NewNode()
    {
        AVLTreeNode* pNode=new AVLTreeNode;
        pNode->m_pLeftSon=nullptr;
        pNode->m_pRightSon=nullptr;
        pNode->m_nHeight=1;
        pNode->m_nValueNumber=0;
        return pNode;
    }
    AVLTreeNode* _NewNode(const _ValyeType& iValue)
    {
        AVLTreeNode* pNode=new AVLTreeNode;
        pNode->m_pLeftSon=nullptr;
        pNode->m_pRightSon=nullptr;
        pNode->m_nHeight=1;
        pNode->m_iValue=iValue;
        pNode->m_nValueNumber=1;
        return pNode;
    }

    int _Height(AVLTreeNode* pNode)
    {
        if(pNode) return pNode->m_nHeight;
        return 0;
    }

    void _PushUp(AVLTreeNode* pNode)
    {
        if(!pNode) return;
        const int nLeftSonHeight=_Height(pNode->m_pLeftSon);
        const int nRightSonHeight=_Height(pNode->m_pRightSon);
        if(nLeftSonHeight<nRightSonHeight) pNode->m_nHeight=1+nRightSonHeight;
        else pNode->m_nHeight=1+nLeftSonHeight;
    }

    /**
       pNode的左孩子将成为根,返回新的树根
    **/
    AVLTreeNode* _LLRotate(AVLTreeNode* pNode)
    {
        if(!pNode) return pNode;
        AVLTreeNode* pLeftSon=pNode->m_pLeftSon;
        pNode->m_pLeftSon=pLeftSon->m_pRightSon;
        pLeftSon->m_pRightSon=pNode;
        _PushUp(pNode);
        _PushUp(pLeftSon);
        return pLeftSon;
    }

    /**
       pNode的右孩子将成为根,返回新的树根
    **/
    AVLTreeNode* _RRRotate(AVLTreeNode* pNode)
    {
        if(!pNode) return pNode;
        AVLTreeNode* pRightSon=pNode->m_pRightSon;
        pNode->m_pRightSon=pRightSon->m_pLeftSon;
        pRightSon->m_pLeftSon=pNode;
        _PushUp(pNode);
        _PushUp(pRightSon);
        return pRightSon;
    }
    /**
        pNode的左孩子的右孩子将成为根,返回新的树根
    **/
    AVLTreeNode* _LRRotate(AVLTreeNode* pNode)
    {
        if(!pNode) return pNode;
        pNode->m_pLeftSon=_RRRotate(pNode->m_pLeftSon);
        return _LLRotate(pNode);
    }

    /**
        pNode的右孩子的左孩子将成为根,返回新的树根
    **/
    AVLTreeNode* _RLRotate(AVLTreeNode* pNode)
    {
        if(!pNode) return pNode;
        pNode->m_pRightSon=_LLRotate(pNode->m_pRightSon);
        return _RRRotate(pNode);
    }

    AVLTreeNode* _Rotate(AVLTreeNode* pNode)
    {
        if(!pNode) return pNode;
        if(2==_Height(pNode->m_pLeftSon)-_Height(pNode->m_pRightSon))
        {
            if(_Height(pNode->m_pLeftSon->m_pLeftSon)>=_Height(pNode->m_pLeftSon->m_pRightSon))
            {
                pNode=_LLRotate(pNode);
            }
            else pNode=_LRRotate(pNode);
        }
        else if(2==_Height(pNode->m_pRightSon)-_Height(pNode->m_pLeftSon))
        {
            if(_Height(pNode->m_pRightSon->m_pLeftSon)>=_Height(pNode->m_pRightSon->m_pRightSon))
            {
                pNode=_RLRotate(pNode);
            }
            else pNode=_RRRotate(pNode);
        }
        return pNode;
    }

    AVLTreeNode* _Insert(AVLTreeNode* pRoot,const _ValyeType& iInsertValue)
    {
        if(nullptr==pRoot)
        {
            pRoot=_NewNode(iInsertValue); return pRoot;
        }
        else if(m_pCompareFunc(iInsertValue,pRoot->m_iValue))
        {
            pRoot->m_pLeftSon=_Insert(pRoot->m_pLeftSon,iInsertValue);
            if(2==_Height(pRoot->m_pLeftSon)-_Height(pRoot->m_pRightSon))
            {
                if(m_pCompareFunc(iInsertValue,pRoot->m_pLeftSon->m_iValue))
                {
                    pRoot=_LLRotate(pRoot);
                }
                else
                {
                    pRoot=_LRRotate(pRoot);
                }
            }
        }
        else if(m_pCompareFunc(pRoot->m_iValue,iInsertValue))
        {
            pRoot->m_pRightSon=_Insert(pRoot->m_pRightSon,iInsertValue);
            if(2==_Height(pRoot->m_pRightSon)-_Height(pRoot->m_pLeftSon))
            {
                if(m_pCompareFunc(iInsertValue,pRoot->m_pRightSon->m_iValue))
                {
                    pRoot=_RLRotate(pRoot);
                }
                else
                {
                    pRoot=_RRRotate(pRoot);
                }
            }
        }
        else
        {
            ++pRoot->m_nValueNumber;
        }

        _PushUp(pRoot);
        return pRoot;
    }

    AVLTreeNode* _Delete(AVLTreeNode* pRoot,const _ValyeType& iDeleteValue)
    {
        if(nullptr==pRoot) return nullptr;
        if(m_pCompareFunc(iDeleteValue,pRoot->m_iValue))
        {
            pRoot->m_pLeftSon=_Delete(pRoot->m_pLeftSon,iDeleteValue);
        }
        else if(m_pCompareFunc(pRoot->m_iValue,iDeleteValue))
        {
            pRoot->m_pRightSon=_Delete(pRoot->m_pRightSon,iDeleteValue);
        }
        else
        {
            if(0==--pRoot->m_nValueNumber)
            {
                if(nullptr==pRoot->m_pLeftSon)
                {
                    AVLTreeNode* pTmp=pRoot;
                    pRoot=pRoot->m_pRightSon;
                    delete pTmp;
                }
                else if(nullptr==pRoot->m_pRightSon)
                {
                    AVLTreeNode* pTmp=pRoot;
                    pRoot=pRoot->m_pLeftSon;
                    delete pTmp;
                }
                else
                {
                    AVLTreeNode* pTmp=pRoot->m_pRightSon;
                    while(pTmp->m_pLeftSon) pTmp=pTmp->m_pLeftSon;
                    pRoot->m_iValue=pTmp->m_iValue;
                    pRoot->m_pRightSon=_Delete(pRoot->m_pRightSon,pRoot->m_iValue);
                }
            }
            else
            {
                return pRoot;
            }
        }
        _PushUp(pRoot);
        if(pRoot&&pRoot->m_pLeftSon) pRoot->m_pLeftSon=_Rotate(pRoot->m_pLeftSon);
        if(pRoot&&pRoot->m_pRightSon) pRoot->m_pRightSon=_Rotate(pRoot->m_pRightSon);
        if(pRoot) pRoot=_Rotate(pRoot);
        return pRoot;
    }

public:
    CAVLTree(_FuncType* pCompareFunc):m_pRoot(nullptr),m_pCompareFunc(pCompareFunc) {}

    void Insert(const _ValyeType& iInsertValue)
    {
        m_pRoot=_Insert(m_pRoot,iInsertValue);
    }

    void Delete(const _ValyeType& iDeleteValue)
    {
        m_pRoot=_Delete(m_pRoot,iDeleteValue);
    }

    int Find(const _ValyeType& iSearchValue)
    {
        AVLTreeNode* pCurrent=m_pRoot;
        while(1)
        {
            if(!pCurrent) break;
            if(m_pCompareFunc(iSearchValue,pCurrent->m_iValue))
            {
                pCurrent=pCurrent->m_pLeftSon;
            }
            else if(m_pCompareFunc(pCurrent->m_iValue,iSearchValue))
            {
                pCurrent=pCurrent->m_pRightSon;
            }
            else return pCurrent->m_nValueNumber;
        }
        return 0;
    }

};
时间: 2024-10-19 02:59:13

AVL学习笔记的相关文章

[平衡树学习笔记]那些你所不知道的鬼畜写法

treap:  (Orz fhq 大神,我曾经以为我会了 treap ,就再也不会写 splay 了,然后我遇上了 lct ) 1 #include <cstdlib> 2 const int sizeOfMemory=10000; 3 template <class type> inline void swap(type & x, type & y) {type t=x; x=y; y=t;} 4 5 namespace treap 6 { 7 struct no

vector 学习笔记

vector 使用练习: /**************************************** * File Name: vector.cpp * Author: sky0917 * Created Time: 2014年04月27日 11:07:33 ****************************************/ #include <iostream> #include <vector> using namespace std; int main

Caliburn.Micro学习笔记(一)----引导类和命名匹配规则

Caliburn.Micro学习笔记(一)----引导类和命名匹配规则 用了几天时间看了一下开源框架Caliburn.Micro 这是他源码的地址http://caliburnmicro.codeplex.com/ 文档也写的很详细,自己在看它的文档和代码时写了一些demo和笔记,还有它实现的原理记录一下 学习Caliburn.Micro要有MEF和MVVM的基础 先说一下他的命名规则和引导类 以后我会把Caliburn.Micro的 Actions IResult,IHandle ICondu

jQuery学习笔记(一):入门

jQuery学习笔记(一):入门 一.JQuery是什么 JQuery是什么?始终是萦绕在我心中的一个问题: 借鉴网上同学们的总结,可以从以下几个方面观察. 不使用JQuery时获取DOM文本的操作如下: 1 document.getElementById('info').value = 'Hello World!'; 使用JQuery时获取DOM文本操作如下: 1 $('#info').val('Hello World!'); 嗯,可以看出,使用JQuery的优势之一是可以使代码更加简练,使开

[原创]java WEB学习笔记93:Hibernate学习之路---Hibernate 缓存介绍,缓存级别,使用二级缓存的情况,二级缓存的架构集合缓存,二级缓存的并发策略,实现步骤,集合缓存,查询缓存,时间戳缓存

本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱好者,互联网技术发烧友 微博:伊直都在0221 QQ:951226918 -----------------------------------------------------------------------------------------------------------------

Activiti 学习笔记记录(三)

上一篇:Activiti 学习笔记记录(二) 导读:上一篇学习了bpmn 画图的常用图形标记.那如何用它们组成一个可用文件呢? 我们知道 bpmn 其实是一个xml 文件

HTML&CSS基础学习笔记8-预格式文本

<pre>标签的主要作用是预格式化文本.被包围在 pre 标签中的文本通常会保留空格和换行符.而文本也会呈现为等宽字体. <pre>标签的一个常见应用就是用来表示计算机的源代码.当然你也可以在你需要在网页中预显示格式时使用它. 会使你的文本换行的标签(例如<h>.<p>)绝不能包含在 <pre> 所定义的块里.尽管有些浏览器会把段落结束标签解释为简单地换行,但是这种行为在所有浏览器上并不都是一样的. 更多学习内容,就在码芽网http://www.

java/android 设计模式学习笔记(14)---外观模式

这篇博客来介绍外观模式(Facade Pattern),外观模式也称为门面模式,它在开发过程中运用频率非常高,尤其是第三方 SDK 基本很大概率都会使用外观模式.通过一个外观类使得整个子系统只有一个统一的高层的接口,这样能够降低用户的使用成本,也对用户屏蔽了很多实现细节.当然,在我们的开发过程中,外观模式也是我们封装 API 的常用手段,例如网络模块.ImageLoader 模块等.其实我们在开发过程中可能已经使用过很多次外观模式,只是没有从理论层面去了解它. 转载请注明出处:http://bl

[原创]java WEB学习笔记48:其他的Servlet 监听器:域对象中属性的变更的事件监听器 (3 个),感知 Session 绑定的事件监听器(2个)

本博客为原创:综合 尚硅谷(http://www.atguigu.com)的系统教程(深表感谢)和 网络上的现有资源(博客,文档,图书等),资源的出处我会标明 本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱好者,互联网技术发烧友 微博:伊直都在0221 QQ:951226918 ---------------------------------