【C++】用红黑树封装map和set

news2024/10/7 14:17:30

我们之前学的map和set在stl源码中都是用红黑树封装实现的,当然,我们也可以模拟来实现一下。在实现之前,我们也可以看一下stl源码是如何实现的。我们上篇博客写的红黑树里面只是一个pair对象,这对于set来说显然是不合适的,所以要想让一个红黑树的代码同时支持map和set,就用上模板就可以了

我们来看看stl源码中是如何实现的

前两个模板参数是两个类型,就是我们要在set或map中放入什么

set不是只需要放入一个吗?所以,set在传参数的时候是这么传的

它的前两个传的全是Key,它这么实现还是为了兼容map,map传的是什么呢?我们再来看一下

传的一个是Key,一个是pair类的对象。那pair中不是已经有Key了吗,为什么还要传Key呢?因为一个最简单的原因之一find函数的参数是Key。

那么看第三个模板参数keyofvalue,传这个类型是为了从value中找到key,因为我们树这个类传给节点类的时候只传了value,如下图:

因为map中value是一个pair对象,set中value就是key,它们的获取方式不一样,所以传这个参数是为了实现仿函数,来取出key值用于比较

那么了解了这个大体的结构之后,下一个就是要实现我们的迭代器了,我们其实可以在红黑树中实现一个树形的迭代器,然后map和set再封装一层就行了,其实我们的迭代器就是一个类,它用来实现类似于指针的一些操作所以我们就用指针来当作这个类的成员变量在这个类的基础上实现迭代器的功能。

在实现迭代器的时候,最关键的一个函数就是重载++,这里迭代器++肯定是按中序,因为这样才有意义,有顺序,那么我们如何通过一个节点找到它的中序遍历的下一个节点呢?这其实是有规律的。比如我们看这样一颗红黑树

首先我们中序遍历是左子树 右子树

1.假设这个节点有右子树,那么这个节点之后就是它的右子树的中序的第一个节点,就是右子树中最左边的节点

2.假设这个节点没有右子树,那么走完这个节点以后以这个节点为根的树就走完了,假如它是它父亲的左孩子,那么就该走它的父亲,如果它是它父亲的右孩子,那么它父亲也走完了,就按照此规律走它的爷爷。

有了这个理论基础,我们就可以来实现了。

同样--的话跟++是完全相反的,反过来的遍历顺序就是右子树,根,左子树,然后我们再分别去看这棵树有没有左子树,如果有,那就走左子树中第一个该走的节点,就是左子树中最右节点;如果没有,那就看它是它父亲的什么节点,一直往上找,直到找到它是它父亲的右子树的节点,它父亲就是下一个要遍历的节点。

下面还有一些细节问题,比如说把迭代器写成模板

那么只需要传不同的类型就可以实现const或非const的迭代器

我们const对象要用const版本的迭代器,因为const对象用普通版本的属于权限放大,所以我们要设计const版本的迭代器

我们也要对红黑树的插入函数进行修改,原来插入函数返回一个bool值,但是库中应该是返回一个pair对象,其中first是个迭代器,second是个bool值表示是否新插入

看到这样的代码的时候,这个typename表示后面是一个类型名,因为static静态成员也可以指明类域然后去访问

另外,我们这里为什么传const K呢?因为就算是普通的迭代器我们也不希望key值改变,因为map的key值改了就不满足二叉搜索树了

这是如何使用const_iterator,首先s就是一个普通的map对象,就调用普通版本的begin()

调完之后它返回一个iterator,而我们用的const_iterator去接收的,所以要写个构造函数,用普通迭代器构造出const迭代器

那么下面我们再整体的来展示一下红黑树和map set之间的封装关系

这就是如何用红黑树封装出map和set,下面是所有的代码

RBTree.h

#include<iostream>
#include<assert.h>
using namespace std;

enum col {
	RED,
	BLACK
};
template<class T>
struct RBTreeNode {
	RBTreeNode(const T& data)
		:_left(nullptr)
		,_right(nullptr)
		,_parent(nullptr)
		,_data(data)
		,_col(RED){}

	RBTreeNode* _left = nullptr;
	RBTreeNode* _right = nullptr;
	RBTreeNode* _parent = nullptr;
	T _data;
	col _col=RED;
};
template<class T,class Ptr,class Ref>
struct RBTreeIterator {
	typedef RBTreeNode<T> Node;
	typedef RBTreeIterator<T,Ptr,Ref> self;

	typedef RBTreeIterator<T,  T*,  T&> iterator;
	typedef RBTreeIterator<T, const T*, const T&> const_iterator;

	Node* _node;

	RBTreeIterator(const iterator& it)
		:_node(it._node) {}

	RBTreeIterator(Node*node)
		:_node(node){}

	Ref operator*() {
		return _node->_data;
	}
	Ptr operator->() {
		return &_node->_data;
	}
	bool operator==(const self&s) {
		return _node == s._node;
	}
	bool operator!=(const self& s) {
		return _node != s._node;
	}
	self& operator++() {
		if (_node == nullptr) {
			cout << "end()不能++" << endl;
			assert(false);
		}
		if (_node->_right) {//有右子树,那么这个节点之后就是它的右子树的中序的第一个节点,就是右子树中最左边的节点
			_node = _node->_right;
			while (_node->_left != nullptr)_node = _node->_left;
			return *this;
		}
		else {//没有右子树,直到找到孩子是父亲左子树的那个父亲节点
			Node* parent = _node->_parent;
			while (parent && _node != parent->_left) {
				parent = parent->_parent;
				_node = _node->_parent;
			}
			_node = parent;
			return *this;
		}
			
	}
	self& operator--() {
		if (_node->_left) {
			_node = _node->_left;
			while (_node->_right != nullptr)_node = _node->_right;
			return *this;
		}
		else {
			Node* parent = _node->_parent;
			while (parent && _node != parent->_right) {
				parent = parent->_parent;
				_node = _node->_parent;
			}
			_node = parent;
			return *this;
		}

	}
};

template<class K,class V,class Keyofvalue>
class RBTree {
	typedef RBTreeNode<V> Node;
public:
	typedef RBTreeIterator<V,V*,V&> iterator;
	typedef RBTreeIterator<V,const V*,const V&> const_iterator;

	const_iterator begin()const {
		Node* cur = _root;
		while (cur && cur->_left)cur = cur->_left;
		return const_iterator(cur);
	}
	iterator begin() {
		Node* cur = _root;
		while (cur&&cur->_left)cur = cur->_left;
		return iterator(cur);
	}
	const_iterator end()const {
		return const_iterator(nullptr);
	}
	iterator end() {
		return iterator(nullptr);
	}
	iterator Find(const K& key) {
		Keyofvalue kov;
		Node* cur = _root;
		while (cur) {
			if (kov(cur->_data) < key) {
				cur = cur->_right;
			}
			else if (kov(cur->_data) > key) {
				cur = cur->_left;
			}
			else {
				return iterator(cur);
			}
		}
		return end();
	}
	pair<iterator,bool> insert(const V& data) {
		if (_root == nullptr) {
			_root = new Node(data);
			_root->_col = BLACK;
			return make_pair(iterator(_root),true);
		}
		Node* cur = _root;
		Node* parent = nullptr;
		Keyofvalue kov;
		while (cur) {
			if (kov(cur->_data) < kov(data)) {
				parent = cur;
				cur = cur->_right;
			}
			else if (kov(cur->_data) > kov(data)) {
				parent = cur;
				cur = cur->_left;
			}
			else return make_pair(iterator(cur),false);
		}
			cur = new Node(data);
			Node* ret = cur;
			if (kov(parent->_data) < kov(cur->_data)) {
				parent->_right = cur;
				cur->_parent = parent;
			}
			else {
				parent->_left = cur;
				cur->_parent = parent;
			}
			Node* c = cur;
			Node* p = cur->_parent;
			Node* g = p->_parent;
			Node* u = nullptr;
			while (p && p->_col == RED) {
				if (p == g->_left)u = g->_right;
				else u = g->_left;
				if (u == nullptr || u->_col == BLACK) {
					if (p == g->_left && c == p->_left) {
						RotateR(g);
						p->_col = BLACK;
						g->_col = RED;
					}
					else if (p == g->_right && c == p->_right) {
						RotateL(g);
						p->_col = BLACK;
						g->_col = RED;
					}
					else if (p == g->_left && c == p->_right) {
						RotateL(p);
						RotateR(g);
						c->_col = BLACK;
						g->_col = RED;
					}
					else if (p == g->_right && c == p->_left) {
						RotateR(p);
						RotateL(g);
						c->_col = BLACK;
						g->_col = RED;
					}
					else assert(false);
					break;
				}
				else if (u->_col == RED) {
					p->_col = BLACK;
					u->_col = BLACK;
					g->_col = RED;
					if (g == _root) {
						g->_col = BLACK;
						break;
					}
					else {
						c = g;
						p = c->_parent;
						g = p->_parent;
					}
				}
				else assert(false);
				
			}
			return make_pair(iterator(ret),true);
	}

	void RotateL(Node* parent) {
		Node* subR = parent->_right;
		Node* subRL = subR->_left;
		Node* ppnode = parent->_parent;
		if (subRL)subRL->_parent = parent;
		parent->_right = subRL;
		subR->_left = parent;
		parent->_parent = subR;
		if (parent == _root) {
			_root = subR;
			subR->_parent = nullptr;
		}
		else {
			subR->_parent = ppnode;
			if (ppnode->_left == parent)ppnode->_left = subR;
			else ppnode->_right = subR;
		}

	}


	void RotateR(Node* parent) {
		Node* subL = parent->_left;
		Node* subLR = subL->_right;
		Node* ppnode = parent->_parent;
		if (subLR)subLR->_parent = parent;
		parent->_left = subLR;
		subL->_right = parent;
		parent->_parent = subL;
		if (parent == _root) {
			_root = subL;
			subL->_parent = nullptr;
		}
		else {
			subL->_parent = ppnode;
			if (ppnode->_left == parent)ppnode->_left = subL;
			else ppnode->_right = subL;
		}
	}
	Node* getroot() {
		return _root;
	}


private:
	Node* _root = nullptr;
};

MySet.h


namespace jxh {
	template<class T>
	class set {
		typedef RBTreeNode<T> Node;
		struct keyofvalue {
			const T& operator()(const T&key) {
				return key;
			}
		};
		void _inorder(Node* root) {
			if (root == nullptr)return;
			_inorder(root->_left);
			cout << root->_data << endl;
			_inorder(root->_right);
		}
	public:
		typedef typename RBTree<T, const T, keyofvalue>::iterator iterator;
		typedef typename RBTree<T, const T, keyofvalue>::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 T& key)
		{
			return _t.insert(key);
		}

		iterator find(const T& key)
		{
			return _t.find(key);
		}

		void inorder() {
			_inorder(_t.getroot());
		}
	private:
		RBTree<T,const T,keyofvalue> _t;
	};

MyMap.h

namespace jxh {
	template<class K,class V>
	class map {
		typedef RBTreeNode<pair<K,V>> Node;
		struct keyofvalue {
			const K& operator()(const pair<K,V>& kv) {
				return kv.first;
			}
		};
		void _inorder(Node* root) {
			if (root == nullptr)return;
			_inorder(root->_left);
			cout << root->_data.first<<" "<<root->_data.second << endl;
			_inorder(root->_right);
		}
	public:

		//typedef RBTreeIterator<pair<K,V>> iterator;
		typedef typename RBTree<K, pair<const K, V>, keyofvalue>::iterator iterator;
		typedef typename RBTree<K, pair<const K, V>, keyofvalue>::const_iterator const_iterator;

		const_iterator begin()const {
			return _t.begin();
		}
		const_iterator end() const{
			return _t.end();
		}
		iterator begin() {
			return _t.begin();
		}
		iterator end() {
			return _t.end();
		}

		pair<iterator, bool> insert(const pair<K, V>& kv)
		{
			return _t.insert(kv);
		}

		iterator find(const K& key)
		{
			return _t.find(key);
		}

		V& operator[](const K& key)
		{
			pair<iterator, bool> ret = insert(make_pair(key, V()));
			return ret.first->second;
		}

		void inorder() {
			_inorder(_t.getroot());
		}
	private:
		RBTree<K, pair<const K,V>, keyofvalue> _t;
	};

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

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

相关文章

软件可靠性基本概念_1.定义和定量描述

1.软件可靠性定义 软件可靠性&#xff08;Software Reliability&#xff09;是软件产品在规定的条件下和规定的时间区间完成规定功能的能力。规定的条件是指直接与软件运行相关的使用该软件的计算机系统的状态和软件的输入条件&#xff0c;或统称为软件运行时的外部输入条件&am…

【MATLAB源码-第183期】基于matlab的图像处理GUI很全面包括滤波,灰度,边缘提取,RGB亮度调节,二值化等。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 1. RGB颜色亮度调整 1.1 RGB颜色模型 RGB颜色模型是一种加色模型&#xff0c;使用红色&#xff08;R&#xff09;、绿色&#xff08;G&#xff09;、蓝色&#xff08;B&#xff09;三种颜色的不同组合来表示各种颜色。每种…

OceanBase V4.2 MySQL模式下,如何通过DBLINK实现跨数据源访问

概述 跨数据源访问可通过 DBLINK&#xff08;以下简称DBLINK&#xff09;实现&#xff0c;从而使得业务代码能够像访问本地数据库一样轻松访问远端数据库。原先&#xff0c;DBLINK主要服务于Oracle模式&#xff0c;但由于OceanBase 的MySQL模式租户同样存在访问远端数据库的需…

【二分查找】Leetcode 点名

题目解析 LCR 173. 点名 算法讲解 1. 哈希表 class Solution { public:int takeAttendance(vector<int>& nums) {map<int, int> Hash;for(auto n : nums) Hash[n];for(int i 0; i < nums[nums.size() - 1]; i){if(Hash[i] 0)return i;}return nums.si…

【ZZULIOJ】1053: 正弦函数(Java)

目录 题目描述 输入 输出 样例输入 Copy 样例输出 Copy code 题目描述 输入x&#xff0c;计算上面公式的前10项和。 输入 输入一个实数x。 输出 输出一个实数&#xff0c;即数列的前10项和&#xff0c;结果保留3位小数。 样例输入 Copy 1 样例输出 Copy 0.841 c…

Python学习笔记11 - 列表

1. 列表的创建与删除 2. 列表的查询操作 3. 列表的增、删、改操作 4. 列表元素的排序 5. 列表生成式

PostgreSQL入门到实战-第十弹

PostgreSQL入门到实战 PostgreSQL数据过滤(三)官网地址PostgreSQL概述PostgreSQL中OR操作理论PostgreSQL中OR实操更新计划 PostgreSQL数据过滤(三) 了解PostgreSQL OR逻辑运算符以及如何使用它来组合多个布尔表达式。 官网地址 声明: 由于操作系统, 版本更新等原因, 文章所列…

免费SSL证书跟付费SSL证书有什么区别?

免费SSL证书与付费SSL证书的主要区别如下&#xff1a; 1. 类型与验证级别&#xff1a; - 免费SSL证书通常仅提供域名验证&#xff08;DV&#xff09;&#xff0c;这是一种最基本的验证级别&#xff0c;仅验证域名的所有权&#xff0c;确认申请者对所申请域名的有效控制。 - 付费…

如何在 iOS 项目中集成 MiniApp SDK,快速构建智能小程序?

本文介绍如何在 iOS 项目中&#xff0c;集成 MiniApp SDK&#xff0c;使之能够构建智能生活小程序&#xff0c;运行在你的 IoT App 上。 准备工作 在集成 MiniApp SDK 之前&#xff0c;您需要在 涂鸦 IoT 开发平台 上&#xff1a; 注册开发者账号、创建产品、创建功能点等。…

汇智知了堂:AIGC引领数字营销新革命,你准备好了吗?

ChatGPT火了&#xff0c;带火的还有AIGC。 经过长时间的蓄力&#xff0c;AIGC技术落地的应用场景呈平铺式展开&#xff0c;已经逐渐渗透到各行各业中&#xff0c;AIGC时代已来。 AIGC浪潮来袭&#xff0c;电商行业从业者该如何应对&#xff1f; AIGC技术会给电商行业带来哪些变…

排序算法—快速排序

文章目录 快速排序一、递归实现二、非递归实现总结 快速排序 以下均以排升序为最终目的。 一、递归实现 有一个排序能解决所有问题吗&#xff1f;没有&#xff01;不过&#xff0c;快速排序这种排序适用于大多数情况。 我们前面讨论排序算法一般都是先讨论一趟的情况&#…

传输层 --- TCP (上篇)

目录 1. TCP 1.1. TCP协议段格式 1.2. TCP的两个问题 1.3. 如何理解可靠性 1.4. 理解确认应答机制 2. TCP 报头中字段的分析 2.1. 序号和确认序号 2.1.1. 序号和确认序号的初步认识 2.1.2. 如何正确理解序号和确认序号 2.2. TCP是如何做到全双工的 2.3. 16位窗口大小…

TypeScript系列之-理解TypeScript类型系统画图讲解

TypeScript的输入输出 如果我们把 Typescript 编译器看成一个黑盒的话。其输入则是使用 TypeScript 语法书写的文本或者文本集合。 输出是编译之后的 JS 文件 和 .d.ts 的声明文件 其中 JS 是将来需要运行的文件(里面是没有ts语法&#xff0c;有一个类型擦除的操作)&#xff0…

Shopee虾皮100%有效提高广告效果的案例分享

Shopee 店铺运营中存在三种广告类型&#xff0c;分别是:关键词广告、关联广告和店铺广告。其中使用最为普遍&#xff0c;主控权最为直接的就是关键词广告&#xff0c;TA的适用范围最广&#xff0c;起效最快&#xff0c;并且可根据自身运营的能力去调控投入产出比&#xff0c;深…

【Python】基础(专版提升1)

Python基础 1. 导学1.1 学习理念1.1.1 弱语法&#xff0c;重本质1.1.2 是技术&#xff0c;更艺术 1.2 学习方法1.2.1 当天知识必须理解 2. Python 简介2.1 计算机基础结构2.1.1 硬件2.1.2 软件 2.2 基础知识2.2.1 Python介绍2.2.1.1定义2.2.1.2优势2.2.1.3从业岗位 2.2.2 Pytho…

数学基础:常见函数图像

来自&#xff1a; https://www.desmos.com/calculator/l3u8133jwj?langzh-CN 推荐: https://www.shuxuele.com/index.html 一、三角函数 1.1 正弦 sin(x) 1.2 余弦 cos(x) 1.3 正切 tan(x) 1.4 余切 cot(x) 1.5 正弦余弦综合 1.6 正切余切综合 二、指数对数

[C语言]——柔性数组

目录 一.柔性数组的特点 二.柔性数组的使用 三.柔性数组的优势 C99中&#xff0c;结构体中的最后⼀个元素允许是未知大小的数组&#xff0c;这就叫做『柔性数组』成员。 typedef struct st_type //typedef可以不写 { int i;int a[0];//柔性数组成员 }type_a; 有些编译器会…

14:00面试,15:00才出来,直接给我问麻了。。

从小厂出来&#xff0c;没想到在另一家公司又寄了。 到这家公司开始上班&#xff0c;加班是每天必不可少的&#xff0c;看在钱给的比较多的份上&#xff0c;就不太计较了。没想到3月一纸通知&#xff0c;所有人不准加班&#xff0c;加班费不仅没有了&#xff0c;薪资还要降40%…

基础语法复习

常用的定义&#xff1a; 读取数据加速&#xff1a; input sys.stdin.readline 设置递归深度&#xff1a; sys.setrecursionlimit(100000) 记忆化搜索&#xff1a; from functools import lru_cache lru_cache(maxsizeNone) 计数器&#xff1a; Counter 类是一个非常有…

Spring Cloud微服务入门(五)

Sentinel的安装与使用 安装部署Sentinel 下载Sentinel&#xff1a; https://github.com/alibaba/Sentinel/releases Sentinel控制台 https://localhost:8080 用户和密码为sentinel 使用Sentinel 加依赖&#xff1a; 写配置&#xff1a; 输入&#xff1a; java -Dserver.po…