• 算法实战:亲自写红黑树之三 算法详解


            此文承接:算法实战:亲自写红黑树之一-CSDN博客

                              算法实战:亲自写红黑树之二 完整代码-CSDN博客

    目录

    一、底层抽象

    二、基本定义

    三、TREE_NODE树节点结构

    四、CRBTree容器结构

    五、结构检查函数

    六、插入的平衡算法

    七、删除的平衡算法


     

    一、底层抽象

            之前已经反复说过,我搞的东西都是在共享内存上运行的,不能使用指针,也就是相当于是在固定大小的数组上操作,实际实现考虑的东西还更多一些,所以对底层做了抽象。

            底层数据节点有一个索引位置,就是数组元素索引,有符号整数,定义为T_SHM_SIZE,-1代表无效。

            从索引到实际位置通过接口转换,具体实现为TREE_NODE的静态成员at(n),这是个关键点。测试代码使用的是静态数组,相关代码如下:

    1. ====================红黑树头文件:
    2. typedef long long T_SHM_SIZE;
    3. //静态数组大小
    4. #define ARRAY_CAPACITY 10000
    5. ====================测试代码文件:
    6. //静态数组
    7. TREE_NODE g_array[ARRAY_CAPACITY];
    8. //从索引到树节点
    9. TREE_NODE& TREE_NODE::at(T_SHM_SIZE n)
    10. {
    11. return g_array[n];
    12. }
    13. //计算树节点自身的索引
    14. T_SHM_SIZE TREE_NODE::_me()const
    15. {
    16. return this - g_array;
    17. }

    二、基本定义

            本代码原来是模板代码,适用于各种不同的数据结构,所以沿用了几个类型定义:

            T_DATA 用户数据类型

            T_COMP 用户数据类型的比较方式,默认是less

            在这个测试代码里面将两个定义实现为:

    1. struct CDemoData
    2. {
    3. long long n = 0;
    4. //用于需要排序的场合
    5. bool operator < (CDemoData const& tmp)const { return n < tmp.n; }
    6. //某些场合也需要等于
    7. bool operator == (CDemoData const& tmp)const { return n == tmp.n; }
    8. friend ostream& operator << (ostream& o, CDemoData const& d)
    9. {
    10. return o << d.n;
    11. }
    12. //用于输出数据的场合
    13. string& toString(string& str)const
    14. {
    15. char buf[2048];
    16. sprintf_s(buf, 2048, "%lld", n);
    17. return str = buf;
    18. }
    19. };
    20. typedef CDemoData T_DATA;
    21. typedef less T_COMP;

            由于相关代码实现的原因,用户数据类型必须按照CDemoData的样式写(也就是模板代码用到了这些功能),不支持简单类型。其实也不是没法支持简单类型,只不过因为实践中不可能在共享内存放简单类型,所以一开始没考虑,后来再修改需要改的地方就多了。

    三、TREE_NODE树节点结构

            终于到了跟树有关的部分,树节点的结构和一般的树结点的结构相比多了删除标志deleted,因为数据是放在数组里面的,删除了位置还在,虽然说删除的节点位于删除链表,无法从树结构访问到,理论上并不需要额外的删除标志,但是为了检查数据结构有这个标志会好很多(由于共享内存的共享性,数据结构被破坏是比较频繁的)。另外由于对齐原因,多一个这个并不会增加树节点大小。

            树节点结构如下:

    1. T_SHM_SIZE hParent;//-1:无,根节点;0-N,子节点,或指向下个空闲地址
    2. T_SHM_SIZE hLeft;//-1表示无子节点
    3. T_SHM_SIZE hRight;//-1表示无子节点
    4. //颜色
    5. bool bColorRed;//是否为红色
    6. //删除标志
    7. signed char deleted;//0:有效,1:删除
    8. T_DATA data;

            树节点有一些内置方法,很容易理解。

    四、CRBTree容器结构

            由于对底层做了抽象,容器内部包含一个数组对象T_SETARRAY,由这个对象提供底层数据,实际用到的就是添加(删除的属于树的删除链表,并不会归还给数组)。

            容器的另外一个数据成员就是TREE_HEAD指针,为什么是指针呢?因为共享内存是独立存在的,大部分操作是连接到已经存在的数据上去,所以是指针,包含的数组对象内部实际也同样是指针。

            在这个测试代码中直接内置了一个TREE_HEAD结构,以便与模板代码保持一致:

    1. private:
    2. TREE_HEAD _tree_head;为了与共享内存操作一致,这个变量不可直接使用,只能使用tree_head
    3. ......
    4. public:
    5. T_SETARRAY m_array;//内置数组对象,存储实际数据
    6. TREE_HEAD* tree_head = &_tree_head;//指向树的头

            TREE_HEAD结构:

    1. struct TREE_HEAD
    2. {
    3. T_SHM_SIZE hHead;
    4. T_SHM_SIZE size;
    5. T_SHM_SIZE free_head;//空闲地址头指针
    6. };

            底层数组T_SETARRAY结构:

    1. struct T_SETARRAY
    2. {
    3. //新版数组头
    4. struct array_head
    5. {
    6. T_SHM_SIZE capacity;
    7. T_SHM_SIZE size;
    8. };
    9. array_head _array_head;这个本来也是在共享内存的,所以不可直接使用,只能使用GetHead()
    10. array_head const* GetHead()const { return &_array_head; }
    11. T_SHM_SIZE capacity()const { return _array_head.capacity; }
    12. T_SHM_SIZE size()const { return _array_head.size; }
    13. T_SHM_SIZE Capacity()const { return _array_head.capacity; }
    14. T_SHM_SIZE Size()const { return _array_head.size; }
    15. struct HANDLE
    16. {
    17. T_SHM_SIZE handle;
    18. };
    19. bool Add(TREE_NODE const& data, HANDLE& h)
    20. {
    21. if (_array_head.size == _array_head.capacity)return false;
    22. else
    23. {
    24. h.handle = _array_head.size;
    25. TREE_NODE::at(h.handle) = data;
    26. ++_array_head.size;
    27. return true;
    28. }
    29. }
    30. };

            由于本代码底层用了静态数组,所以这个结构修改了内部实现,仅仅保持接口和原来相同。这个结构与算法基本没什么关系,不用太关注。

            容器还定义了迭代器iterator,符合一般的规则,代码也很简单。

            容器初始化很简单:

    1. CRBTree() :m_OldValueSeted(false)
    2. {
    3. m_array._array_head.capacity = ARRAY_CAPACITY;
    4. m_array._array_head.size = 0;
    5. }

            初始化了数组的容量和大小。如果是共享内存,这些数据就要从共享内存获得。当然,这跟算法没什么关系。

    五、结构检查函数

            有一组结构检查函数,用于检查数据是否正确。

    • bool _check_handle(T_SHM_SIZE h)const 检查h是否合法,必须大于等于-1,注意,不是检查是否是数据节点
    • bool _check_is_data_node(T_SHM_SIZE h)const 这才是检查是否是数据节点,不能是-1
    • T_SHM_SIZE _check_get_count(T_SHM_SIZE h) 获取节点总数,包括自身
    • void _check_show_tree(......) 显示树形结构
    • void debug() 无参数,显示整个树形结构
    • bool _check_rbtree() 检查红黑树特征,主要是调用下一个函数
    • bool _check_rbtree_count(......) 递归检测红黑树特征
    • bool _check()const 检查普通树特征,先调用这个检查树结构,再调用_check_rbtree检查红黑树特征

    六、插入的平衡算法

            待续

    七、删除的平衡算法

            待续

    (这里是结束)

  • 相关阅读:
    小学生python游戏编程arcade----敌人精灵上方显示方框及子弹显示问题
    VSCode 环境配置原理
    第六节:数组的定义与使用【java】
    jsp 九大内置对象和四大域对象
    idea 创建java web项目 run后出现404现象
    openGauss学习笔记-70 openGauss 数据库管理-创建和管理普通表-查看表数据
    Android 音频开发入门指南
    R语言拟合ARIMA模型并使用拟合模型进行预测推理、使用autoplot函数可视化ARIMA模型预测结果、可视化包含置信区间的预测结果
    【VSCode】对比两个文件差异
    SAP ABAP CO02/COHV工单下达自动生成客制化批次
  • 原文地址:https://blog.csdn.net/2301_77171572/article/details/134423144