STL容器之map和set的补充map与set的封装

一、模拟实现map和set

1.1三个类模板的设计

template <class Key, class Value, class KeyOfValue, class Compare,class Alloc = alloc>class rb_tree{};//红黑树类模板template <class Key, class T, class Compare = less<Key>, class Alloc = alloc>
class map{};//map对红黑树进行了封装
template <class Key, class Compare = less<Key>, class Alloc = alloc>
class set{};//set对红黑树进行了封装

​ 实际上红黑树类模板的第二个模板参数决定了红黑树的节点用来存放key还是pair;第一个模板参数是因为find之类的函数需要传递key,而第二个模板参数可能是pair无法实现实现直接使用key进行比较;第三个模板参数是为了解决insert是需要根据键值来进行搜索二叉树的建立,但是传递的参数是data,可能是键值,也可能是pair,所以需要使用仿函数,将key取出来,主要是为了map设计的;

​ 为什么不使用仿函数直接比较而是返回key值,因为仿函数是最外层进行传入的,最外层传递比较方式,而内层的红黑树关心的是键值比较,所以红黑树使用KeyOfT;

1.2迭代器的设计

在rbtree中设计。

1.2.1begin/end

​ begin返回值是最左节点,end返回的是nullptr;

​ 而库里面实现的是增加了一个哨兵位,使得哨兵位的左指向最小值,右指向最大值,end是哨兵位;好处是获取begin较快,但是删除和插入的成本增加了;

1.2.2++和- -的设计

​ ++走中序遍历;

​ 1.右不为空,访问右树的最左节点;

​ 2.右为空,访问满足孩子是父亲的左孩子的最近祖先;

​ - - 反的走中序遍历

​ 1.左不为空,访问左树的最右节点;

​ 2.左为空,访问满足孩子是父亲的右孩子的最近祖先;

1.2.3const迭代器

​ RBTree通过模板参数实现const的迭代器类,set封装使用const迭代器来定义普通和const迭代器,即set使用的迭代器都是const迭代器;map使用pair<const K,V>来实现普通的迭代器不可以访问key;

1.2.4反向迭代器

1.3插入实现

​ 注意返回值是pair,要注意对于set返回值要使用const迭代器的构造(普通迭代器对const迭代器的构造),因为普通迭代器和const迭代器会发生冲突;

​ 对于tree迭代器类的设计其实是设计了一个迭代器的拷贝构造函数,set调用时变成普通迭代器对const迭代器的构造;然后tree迭代器内嵌了普通迭代器的类型。对于内嵌类型类外使用时不会给提示的,还必须typename修饰。

1.4operator[]

​ 使用insert实现。

1.5map,tree,set的代码实现

namespace RbTree
{enum Colour{RED,BLACK,};template <class T>struct RBTreeNode{RBTreeNode(const T &data) : data_(data), left_(nullptr), right_(nullptr), parent_(nullptr), col_(RED) {}T data_;RBTreeNode<T> *left_;RBTreeNode<T> *right_;RBTreeNode<T> *parent_;Colour col_;};template <class T, class Ptr, class Ref>struct __TreeIterator{typedef RBTreeNode<T> node;typedef __TreeIterator<T, Ptr, Ref> Self;typedef __TreeIterator<T, T *, T &> iterator;__TreeIterator(node *node = nullptr) : node_(node) {}__TreeIterator(const iterator &it) : node_(nullptr){node_ = it.node_;}Ref operator*(){return node_->data_;}Ptr operator->(){return &(node_->data_);}Self &operator++(){if (node_->right_){// 右不为空,访问右树的最左节点node *subleft = node_->right_;while (subleft->left_){subleft = subleft->left_;}node_ = subleft;}else{// 右为空,访问满足孩子是父亲的左孩子的最近的祖先node *parent = node_->parent_;node *cur = node_;while (parent && cur != parent->left_){cur = parent;parent = parent->parent_;}// 判断是否找到祖先节点if (!parent){node_ = nullptr;}else{node_ = parent;}}return *this;}Self &operator--(){ // 右子树 根 左子树if (node_->left_){node *rightmost = node_->left_;while (rightmost->right_){rightmost = rightmost->right_;}node_ = rightmost;}else{node *parent = node_->parent_;node *cur = node_;while (parent && cur != parent->right_){cur = parent;parent = parent->parent_;}// 判断是否找到祖先节点if (!parent){node_ = nullptr;}else{node_ = parent;}}return *this;}bool operator!=(const Self &iterator) const{return node_ != iterator.node_;}bool operator==(const Self &iterator) const{return node_ == iterator.node_;}node *node_;};template <class K, class T, class KeyOfT>class RBTree{typedef RBTreeNode<T> node;public:typedef __TreeIterator<T, T *, T &> iterator;typedef __TreeIterator<T, const T *, const T &> const_iterator;iterator begin(){node *leftmost = root_;while (leftmost && leftmost->left_){leftmost = leftmost->left_;}return leftmost;}iterator end(){return nullptr;}const_iterator begin() const{node *leftmost = root_;while (leftmost && leftmost->left_){leftmost = leftmost->left_;}return leftmost;}const_iterator end() const{return nullptr;}node *find(const K &key){KeyOfT kot;node *cur = root_;while (cur){if (kot(cur->data_) < key){cur = cur->right_;}else if (kot(cur->data_) > key){cur = cur->left_;}else{return cur;}}return nullptr;}std::pair<iterator, bool> insert(const T &data);void rotatel(node *parent);void rotater(node *parent);void rotaterl(node *parent);void rotatelr(node *parent);void inorder(){_inorder(root_);}bool isrbtree(){return _isrbtree(root_);}bool checkcolour(node *root, int blacknum, int reference);private:void _inorder(node *root);bool _isrbtree(node *root);node *root_ = nullptr;};template <class K, class T, class KeyOfT>void RBTree<K, T, KeyOfT>::rotatel(node *parent){node *cur = parent->right_;node *curleft = cur->left_;parent->right_ = curleft;node *ppnode = parent->parent_;if (curleft)curleft->parent_ = parent;parent->parent_ = cur;cur->left_ = parent;if (parent == root_){root_ = cur;cur->parent_ = nullptr;}else{if (ppnode->left_ == parent){ppnode->left_ = cur;}else{ppnode->right_ = cur;}cur->parent_ = ppnode;}}template <class K, class T, class KeyOfT>void RBTree<K, T, KeyOfT>::rotater(node *parent){node *cur = parent->left_;node *curright = cur->right_;node *ppnode = parent->parent_;parent->left_ = curright;cur->right_ = parent;if (curright)curright->parent_ = parent;parent->parent_ = cur;if (parent == root_){root_ = cur;cur->parent_ = nullptr;}else{if (ppnode->left_ == parent){ppnode->left_ = cur;}else{ppnode->right_ = cur;}cur->parent_ = ppnode;}}template <class K, class T, class KeyOfT>void RBTree<K, T, KeyOfT>::rotaterl(node *parent){rotater(parent->right_);rotatel(parent);}template <class K, class T, class KeyOfT>void RBTree<K, T, KeyOfT>::rotatelr(node *parent){rotatel(parent->left_);rotater(parent);}template <class K, class T, class KeyOfT>pair<typename RBTree<K, T, KeyOfT>::iterator, bool> RBTree<K, T, KeyOfT>::insert(const T &data){if (root_ == nullptr){root_ = new node(data);root_->col_ = BLACK;return {root_, true};}node *cur = root_;node *parent = cur;KeyOfT kot;while (cur){if (kot(data) > kot(cur->data_)){parent = cur;cur = cur->right_;}else if (kot(data) < kot(cur->data_)){parent = cur;cur = cur->left_;}else{return {cur, false};}}cur = new node(data);node *newnode = cur;cur->col_ = RED;if (kot(data) > kot(parent->data_)){parent->right_ = cur;}else{parent->left_ = cur;}cur->parent_ = parent;while (parent && parent->col_ == RED){node *grandfather = parent->parent_;if (grandfather->left_ == parent){node *uncle = grandfather->right_;// 叔叔存在且为红if (uncle && uncle->col_ == RED){// 变色+向上处理parent->col_ = uncle->col_ = BLACK;grandfather->col_ = RED;// 继续向上处理cur = grandfather;parent = cur->parent_;}else{// 叔叔不存在或者叔叔存在且为黑if (cur == parent->left_){// 右单旋rotater(grandfather);parent->col_ = BLACK;grandfather->col_ = RED;}else{// 左右双旋rotatelr(grandfather);cur->col_ = BLACK;grandfather->col_ = RED;}break;}}else if (grandfather->right_ == parent){node *uncle = grandfather->left_;// 叔叔存在且为红if (uncle && uncle->col_ == RED){// 变色+向上处理parent->col_ = uncle->col_ = BLACK;grandfather->col_ = RED;// 继续向上处理}else{// 叔叔不存在或者叔叔存在且为黑if (cur == parent->right_){// 左单旋rotatel(grandfather);parent->col_ = BLACK;grandfather->col_ = RED;}else{// 右左双旋rotaterl(grandfather);cur->col_ = BLACK;grandfather->col_ = RED;}break;}}else{assert(false);}}// 如果父亲不存在,说明cur为根,就要将cur置为黑色,如果父亲存在且为黑色则不需要处理;root_->col_ = BLACK;return {newnode, true};}template <class K, class V, class KeyOfT>void RBTree<K, V, KeyOfT>::_inorder(node *root){if (root == nullptr){return;}_inorder(root->left_);std::cout << root->data_;_inorder(root->right_);}template <class K, class V, class KeyOfT>bool RBTree<K, V, KeyOfT>::checkcolour(node *root, int blacknum, int reference){if (root == nullptr){if (blacknum != reference){return false;}return true;}if (root->col_ == BLACK){blacknum++;}if (root->col_ == RED && root->parent_->col_ == RED){KeyOfT kot;std::cout << kot(root->data_) << "出现连续红色节点" << std::endl;return false;}return checkcolour(root->left_, blacknum, reference) && checkcolour(root->right_, blacknum, reference);}template <class K, class V, class KeyOfT>bool RBTree<K, V, KeyOfT>::_isrbtree(node *root){if (root == nullptr){return true;}if (root->col_ != BLACK){return false;}int reference = 0;node *cur = root_;while (cur){if (cur->col_ == BLACK){reference++;}cur = cur->left_;}return checkcolour(root, 0, reference); // 检查连续的红色}
}namespace Map
{template <class K, class V>class map{struct MapKeyOfT{const K &operator()(const pair<const K, V> &kv){return kv.first;}};public:typedef typename RbTree::RBTree<K, pair<const K, V>, MapKeyOfT>::iterator iterator;typedef typename RbTree::RBTree<K, pair<const K, V>, MapKeyOfT>::const_iterator const_iterator;iterator begin(){return t_.begin();}iterator end(){return t_.end();}const_iterator begin() const{return t_.begin();}const_iterator end() const{return t_.end();}pair<iterator, bool> insert(const pair<const K, V> &kv){return t_.insert(kv);}V &operator[](const K &key){pair<iterator, bool> ret = insert({key, V()});return ret.first->second;}private:RbTree::RBTree<K, pair<const K, V>, MapKeyOfT> t_;};
}namespace Set
{template <class K>class set{struct SetKeyOfT{const K& operator()(const K& key){return key;}};public:typedef typename RbTree::RBTree<K, K, SetKeyOfT>::const_iterator iterator;typedef typename RbTree::RBTree<K, K, SetKeyOfT>::const_iterator const_iterator;iterator begin(){return t_.begin();}iterator end(){return t_.end();}const_iterator begin() const{return t_.begin();}const_iterator end() const{return t_.end();}pair<iterator, bool> insert(const K& key){pair<typename RbTree::RBTree<K, K, SetKeyOfT>::iterator, bool> ret = t_.insert(key);return ret;}private:RbTree::RBTree<K, K, SetKeyOfT> t_;};
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/727848.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Mybatis-Plus Mapper映射文件使用

介绍 MyBatis 的真正强大在于它的语句映射&#xff0c;这是它的魔力所在。由于它的异常强大&#xff0c;映射器的 XML 文件就显得相对简单。如果拿它跟具有相同功能的 JDBC 代码进行对比&#xff0c;你会立即发现省掉了将近 95% 的代码。MyBatis 致力于减少使用成本&#xff0…

国外文献搜索免费下载网站

文献免费下载路径&#xff1a; sci-hub&#xff1a; https://www.sci-hub.ren/ Sci-Hub 最初是一种提供快速访问科学期刊文章的工具 - 此类文章是当今传播科学知识的主要媒介。现在&#xff0c;Sci-Hub 已经建立了一个包含88,343,822篇研究文章和书籍的数据库&#xff0c;任何…

day2 FreeRTOS

思维导图 1.使用ADC采样光敏电阻数值&#xff0c;如何根据这个数值调节LED灯亮度。 HAL_TIM_PWM_Start(&htim3,TIM_CHANNEL_3);//打开定时器3的通道3并且设置为PWM功能/* USER CODE END 2 *//* Infinite loop *//* USER CODE BEGIN WHILE */int adc_val 0;while (1){HAL_…

Day15:技术架构、Maven、Spring Initializer、Spring全家桶、Spring IoC

侧重于服务端&#xff08;后端&#xff09;&#xff0c;不在意前端&#xff0c;了解一些前端即可&#xff09; 技术架构 &#xff08;把Spring设计的更简单好用了就是Spring Boot&#xff09; 开发环境&#xff08;Maven&#xff09; Maven maven通过brew安装的目录为&#x…

开放式高实时高性能PLC控制器解决方案-基于米尔电子STM32MP135

前言 随着工业数字化进程加速与IT/OT深入融合&#xff0c;不断增加的OT核心数据已经逐步成为工业自动化行业的核心资产&#xff0c;而OT层数据具备高实时、高精度、冗余度高、数据量大等等特点&#xff0c;如何获取更加精准的OT数据对数字化进程起到至关重要的作用&#xff0c;…

Unity类银河恶魔城学习记录8-4 P80 Blackhole ability state源代码

Alex教程每一P的教程原代码加上我自己的理解初步理解写的注释&#xff0c;可供学习Alex教程的人参考 此代码仅为较上一P有所改变的代码 【Unity教程】从0编程制作类银河恶魔城游戏_哔哩哔哩_bilibili Entity.cs using System.Collections; using System.Collections.Generic…

STM32CubeMX学习笔记15---CAN总线

1、CAN简介 CAN总线网络的结构有闭环和开环两种形式 闭环结构的CAN总线网络&#xff0c;总线两端各连接一个1202的电阻。这种CAN总线网络由ISO11898标准定义&#xff0c;是高速、短距离的CAN网络&#xff0c;通信速率为125kbit/s到1Mbit/s。在1Mbit/s通信速率时&#x…

基于STC系列单片机实现PNP型三极管S8550驱动共阳数码管或NPN型三极管S8050驱动共阴数码管功能

Digitron.c #include "Digitron.h" //#include "Key.h" #define uchar unsigned char//自定义无符号字符型为uchar #define uint unsigned int//自定义无符号整数型为uint //uchar code DigitronBitCodeArray[] = {0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x…

牛客每日一题之 前缀和

目录 题目介绍&#xff1a; 算法原理&#xff1a; 前缀和&#xff1a; 代码实现&#xff1a; 题目介绍&#xff1a; 题目链接&#xff1a;【模板】前缀和_牛客题霸_牛客网 算法原理&#xff1a; 先讲讲暴力解法每次求出数组下标r之前元素的和&#xff0c;再减去数组下标l-…

Java特性之设计模式【过滤器模式】

一、过滤器模式 概述 ​ 过滤器模式&#xff08;Filter Pattern&#xff09;或标准模式&#xff08;Criteria Pattern&#xff09;是一种设计模式&#xff0c;这种模式允许开发人员使用不同的标准来过滤一组对象&#xff0c;通过逻辑运算以解耦的方式把它们连接起来。这种类型的…

OpenStack之Nova

一 、Nova 使用OpenStack Compute来托管和管理云计算系统。 OpenStack Compute是基础架构即服务 &#xff08;IaaS&#xff09;系统的主要部分。 主要模块在Python中实现&#xff1a; 1因为认证&#xff0c;与OpenStack 身份认证keystone 交互。 2因为磁盘和服务器镜像&#xf…

【Maven】Maven 基础教程(五): jar 包冲突问题

《Maven 基础教程》系列&#xff0c;包含以下 5 篇文章&#xff1a; Maven 基础教程&#xff08;一&#xff09;&#xff1a;基础介绍、开发环境配置Maven 基础教程&#xff08;二&#xff09;&#xff1a;Maven 的使用Maven 基础教程&#xff08;三&#xff09;&#xff1a;b…

C++ · 代码笔记3 · 引用

目录 前言011引用初探_引用与普通变量012引用初探_引用作为函数参数013引用初探_引用作为函数返回值014引用初探_引用返回局部函数造成的错误015引用初探_多级引用020引用与指针递增的区别030const与引用040使用const限定的函数形参引用 前言 本笔记所涉及到的编程环境与 《C …

经典语义分割(二)医学图像分割模型UNet

经典语义分割(二)医学图像分割模型UNet 我们之前介绍了全卷积神经网络( FCN) &#xff0c;FCN是基于深度学习的语义分割算法的开山之作。 今天我们介绍另一个语义分割的经典模型—UNet&#xff0c;它兼具轻量化与高性能&#xff0c;通常作为语义分割任务的基线测试模型&#x…

opengl 学习(三)-----着色器

着色器 分类demo效果解析教程 分类 OPengl C demo #include "glad/glad.h" #include "glfw3.h" #include <iostream> #include <cmath> #include <vector>#include <string> #include <fstream> #include <sstream>…

苍穹外卖技术栈

Day5 Redis_Spring Data Redis使用方法 Spring Data Redis Spring Date Redis 是Spring的一部分&#xff0c; 对Redis底层开发包进行了高度封装&#xff0c;在Spring项目中&#xff0c;可以使用Spring Data Redis来简化操作。 操作步骤 导入Spring Data Redis 的maven坐标配置…

浮点数和定点数

前言 大家好我是jiantaoyab&#xff0c;这是我所总结作为学习的笔记第七篇,在这里分享给大家,还有一些书籍《深入理解计算机系统》《计算机组成&#xff1a;结构化方法》《计算机体系结构&#xff1a;量化研究方法》&#xff0c;今天我们来了解定点数和浮点数 定点数 BCD编码 …

JavaScript基础4之原型的原型继承、原型链和理解对象的数据属性、访问器属性

JavaScript基础 原型原型继承问题解决 原型链isPrototypeOf()Object.getPrototypeOf() 理解对象数据属性访问器属性 原型 原型继承 继承是面向对象编程的另一个特征&#xff0c;通过继承进一步提升代码封装的程度&#xff0c;JavaScript中大多是借助原型对象实现继承的特性。…

蜘蛛池是什么意思,怎么生成蜘蛛池

蜘蛛池是由自然界中的蜘蛛群落构成的一个小生态系统&#xff0c;也是身处自然界中的游客们可以在风雨中体验到最贴近自然气息的地方。 点开我主页面 Baidu蜘蛛的作用&#xff1a; 引蜘蛛逐渐收录&#xff0c;降权引蜘蛛可以疗伤&#xff0c;排名/收录不稳定&#xff0c;没有收…

等离子体密度诊断集

文章目录 等离子体密度诊断电子密度诊断朗缪尔探针法 (Langmuir Probe)干涉度量法 (Interferometer)全息法 (Holographic Method)离子密度诊断光腔衰减震荡 (Cavity Ring-Down Spectroscopy, CRDS)发射光谱法 (OES)中性粒子密度吸收光谱法 (Absorption Spectroscopy, AS)发射光…