//---------------------------15/03/22----------------------------

//一直好奇KeyOfValue是什么,查了下就是一个和仿函数差不多的东西,在第7章会详细介绍

//现在只知道KeyOfValue()可以构造一个类调用他的operator()可以得到一个value的key

//允许重复的插入

template<class Key,class Value,
class KeyOfValue,class Compare,
class Alloc>

typename rb_tree<key, Value, KeyOfValue, Compare, Alloc>::iterator

rb_tree<Key, Value, KeyOfValue, Compare, Alloc>::insert_equal(const Value& v)

{

link_type y = header;

link_type x = root();

//从根节点开始搜索直到x为叶子节点

while ( x !=
)

{

y = x;

x = key_compare(KeyOfValue()(v),key(x)) ? left(x) : right(x);

}

//把一个值为v的节点插入x的位置,y是x的父节点

return __insert(x,y,v);

}

//不允许重复

//可以插入(没有重复)返回的bool为true,否则为false

template<class Key,class Value,
class KeyOfValue,class Compare,
class Alloc>

pair<typename rb_tree<Key, Value, KeyOfValue, Compare, Alloc>::iterator,bool>

rb_tree<Key, Value, KeyOfValue, Compare, Alloc>::insert_unique(const Value& v)

{

link_type y = header;

link_type x =root();

bool comp =
true;

while( x !=
)

{

y = x;

comp = key_compare(KeyOfValue()(v),key(x));

x = comp ? left(x) :right(x);

}

iterator j = iterator(y);

//最后如果v "小于"(假设比较操作是小于) y

//这边做这样的操作是因为"大于等于"的时候都会向右走,所以需要换一下j.node和v的位置

//看看是否不相等

if(comp)

if(j == begin())   
//如果插入点的父节点是最左边的节点

return pair<iterator,bool>(__insert(x,y,v),true);

else           
//不是最左边就要--j继续判断

--j;

//这是因为只有最左边可以确定没有比v小的元素了

//如果j.node确实小于v就可以插入。

if(key_compare(key(j.node),KeyOfValue()(v)))

return pair<iterator,bool>(__insert(x,y,v),true);

return pair<iterator,bool>(j,false);

/*

总结下:会出现想等的情况只有有向右走的情况

如果最后是向右走的,那么有可能相等的元素就是j也就是y也就是要插入位置
的父节点

如果最后是向左走的,那么有可能相等的就是比j小的最大的那个元素,也就是--j

*/

}

//__insert

template<class Key,class Value,
class KeyOfValue,class Compare,
class Alloc>

typename    rb_tree<key, Value, KeyOfValue, Compare, Alloc>::iterator

rb_tree<Key, Value, KeyOfValue, Compare, Alloc>::

__insert(base_ptr x_, base_ptr y_,const Value& v)

{

link_type x = (link_type) x_;

link_type y = (link_type) y_;

link_type z;

//如果要插入的是根节点或者
要插入的位置不是空(header的左右儿子) 或v小于y 那么就是要插入到左边

if(y == header || x !=
|| key_compare(KeyOfValue()(v),key(y)))

{

z = create_node(v);

left(y) = z;   //如果y是header那可以设置leftmost为z

//如果要插入的是根节点,

if(y == header)

{

//设置根节点

root() = z;

//设置最右边的元素

rightmost() = z;

}

//如果要插到最左边

else
if(y == leftmost())

leftmost() = z;

}

else

{

z = create_node(v);

right(y) = z;

if (y == rightmost())

rightmost() = z;

}

//父节点为y左右儿子都是
空。

parent(z) = y;

left(z) =;

right(z) =;

//调整平衡

__rb_tree_rebalance(z,header->parent);

++node_count;

return iterator(z);

}

//旋转以及变色

//平衡这里inline是什么意思?里面明明有循环
而且代码这么长

//下面的代码几乎和算法导论里的一样

inline
void

__rb_tree_rebalance(__rb_tree_node_base* x, __rb_tree_node_base*& root)

{

x->color = __rb_tree_red;

while(x != root && x->parent->color == __rb_tree_red)

{

if(x->parent == x->parent->parent->left)

{

__rb_tree_node_base* y =x->parent->parent->right;

if(y && y->color == __rb_tree_red)

{

x->parent->color = __rb_tree_black;

y->color = __rb_tree_black;

x->parent->parent->color = __rb_tree_red;

x = x->parent->parent;

}

else

{

if(x == x->parent->right)

{

x = x->parent;

__rb_tree_rotate_left(x, root);

}

x->parent->color = __rb_tree_black;

x->parent->parent->color = __rb_tree_red;

__rb_tree_rotate_right(x->parent->parent, root);

}

}

else

{

__rb_tree_node_base* y =x->parent->parent->left;

if(y && y->color == __rb_tree_red)

{

x->parent->color = __rb_tree_black;

y->color = __rb_tree_black;

x->parent->parent->color = __rb_tree_red;

x = x->parent->parent;

}

else

{

if(x == x->parent->left)

{

x = x->parent;

__rb_tree_rotate_right(x,root);

}

x->parent->color = __rb_tree_black;

x->parent->parent->color = __rb_tree_red;

__rb_tree_rotate_left(x->parent->parent,root);

}

}

}

root->color = __rb_tree_black;

}

//左转

inline voide

__rb_tree_rotate_left(__rb_tree_node_base* x,__rb_tree_node_base*& root)

{

__rb_tree_node_base* y = x->right;

x->right = y->left;

if(y->left !=
)

y->left->parent = x;

y->parent = x->parent;

if(x == root)

root = y;

else
if (x == x->parent->left)

x->parent->left = y;

else

x->parent->right = y;

y->left = x;

y->parent = y;

}

//右转

inline voide

__rb_tree_rotate_right(__rb_tree_node_base* x,__rb_tree_node_base*& root)

{

__rb_tree_node_base* y = x->left;

x->left = y->right;

if(y->right !=
)

y->right->parent = x;

y->parent = x->parent;

if(x == root)

root = y;

else
if (x == x->parent->right)

x->parent->right = y;

else

x->parent->left = y;

y->right= x;

y->parent = y;

}

//find

template<class Key,class Value,
class KeyOfValue,class Compare,
class Alloc>

typename    rb_tree<key, Value, KeyOfValue, Compare, Alloc>::iterator

rb_tree<Key, Value, KeyOfValue, Compare, Alloc>::find(const Key& k)

{

link_type y =header;

link_type x =root();

while(x !=
)

if(!key_compare(key(x), k))

y = x, x =left(x);

else

x = right(x);

iterator j =iterator(y);

return (j ==end() || key_compare(k, key(j.node))) ? end() :j;

}

//最后没有delete操作,之前的算法导论部分已经给出了delete操作


stl源码剖析 详细学习笔记 RB_tree (2)的更多相关文章

  1. stl源码剖析 详细学习笔记 RB_tree (1)

    // //  RB_tree_STL.cpp //  笔记 // //  Created by fam on 15/3/21. // // #include "RB_tree_STL.h&q ...

  2. stl源码剖析 详细学习笔记 set map

    // //  set map.cpp //  笔记 // //  Created by fam on 15/3/23. // // //---------------------------15/03 ...

  3. stl源码剖析 详细学习笔记 hashtable

    //---------------------------15/03/24---------------------------- //hashtable { /* 概述: sgi采用的是开链法完成h ...

  4. stl源码剖析 详细学习笔记heap

    // //  heap.cpp //  笔记 // //  Created by fam on 15/3/15. // // //---------------------------15/03/15 ...

  5. stl源码剖析 详细学习笔记 空间配置器

    //---------------------------15/04/05---------------------------- /* 空间配置器概述: 1:new操作包含两个阶段操作 1>调 ...

  6. stl源码剖析 详细学习笔记 算法(1)

    //---------------------------15/03/27---------------------------- //算法 { /* 质变算法:会改变操作对象之值 所有的stl算法都 ...

  7. stl源码剖析 详细学习笔记 算法总览

    //****************************基本算法***************************** /* stl算法总览,不在stl标准规格的sgi专属算法,都以 *加以标 ...

  8. stl源码剖析 详细学习笔记 hashset hashmap

    //---------------------------15/03/26---------------------------- //hash_set { /* hash_set概述: 1:这是一个 ...

  9. stl源码剖析 详细学习笔记priority_queue slist

    // //  priority_queue.cpp //  笔记 // //  Created by fam on 15/3/16. // // //------------------------- ...

随机推荐

  1. Oracle EBS FTP显示无法与某IP 连接

    首先 用root用户登录 如果可以登录 那么应该是权限的问题 这里选择 方法二:修改 /etc/sudoers 文件,找到下面一行,在root下面添加一行,如下所示: ## Allow root to ...

  2. MySQL-死锁查询

    1.查询是否锁表 show OPEN TABLES where In_use > 0; 查询到相对应的进程 === 然后 kill    id 2.查询进程 show processlist 补 ...

  3. ASP.NET动态引用样式表(css)和脚本(js)文件

    // 引入js文件 HtmlGenericControl scriptControl = new HtmlGenericControl("script"); scriptContr ...

  4. Docker容器学习与分享05

    Docker镜像操作 学完了一些最基本的操作之后,我学习了一些关于docker镜像的基本操作. 首先来学习一下从docker hub上拉取镜像,以centos镜像为例,使用docker search命 ...

  5. 团队作业——Alpha冲刺 9/12

    团队作业--Alpha冲刺 冲刺任务安排 杨光海天 今日任务:修复编辑界面与弹窗界面合并中出现的BUG 明日任务:希望完成编辑界面所有接口交互的功能 郭剑南 今日任务:优化图像预处理所有功能的函数代码 ...

  6. PyQt5--QFileDiaglog

    # -*- coding:utf-8 -*- ''' Created on Sep 17, 2018 @author: SaShuangYiBing Comment: ''' import sys f ...

  7. JFreeChart框架中生成饼状图上怎样显示数据 [问题点数:40分,结帖人GreenLawn]

    我用JFreeChart框架生成饼状图,但想把数据信息在饼图上显示,是在饼图内部(即圆内)显示!怎样实现啊??  去掉lablepieplot.setLabelGenerator(null);去掉线p ...

  8. Python基础5

    本节内容 1. 函数基本语法及特性 2. 参数与局部变量 3. 返回值 嵌套函数 4.递归 5.匿名函数 6.函数式编程介绍 7.高阶函数 8.内置函数 温故知新 1. 集合 主要作用: 去重 关系测 ...

  9. 【[AHOI2013]差异】

    这个题一看就是为后缀家族设计的 我们看到我们要求的这个柿子 \[\sum_{i=1}^n\sum_{j=i+1}^nT_i+T_j-2\times lcp(T_i,T_j)\] 显然的是前面的那些东西 ...

  10. 3532: [Sdoi2014]Lis

    Description 给定序列A,序列中的每一项Ai有删除代价Bi和附加属性Ci.请删除若干项,使得4的最长上升子序列长度减少至少1,且付出的代价之和最小,并输出方案. 如果有多种方案,请输出将删去 ...