当前位置: 首页 > news >正文

C++--哈希封装my_unordered_set和my_unordered_map

目录

一,引言

二,基本结构

三,hash迭代器

四,HashTable的基本结构


一,引言

在实现哈希表之后,在unordered_set和unordered_map的学习中。了解到这两者的数据结构底层是由哈希表实现的,为此在实现哈希表的基础上对哈希表进行封装。

二,基本结构

首先要需要对Node节点进行修改。此过程和红黑树的封装类似。由上层结构决定是set还是map:
参考代码如下:

template<class T> struct HashNode { T _data; HashNode<T>* _next; HashNode(const T& data) :_data(data) , _next(nullptr) {} };

T的具体类型由上层传入进行决定。


KeyOFT仿函数以及hash仿函数

在上述节点的基本类型中为T类型,因此并不直到上层究竟是map还是set。为此第一个仿函数的作用是返回这两者的key值。

hash仿函数和哈希表的实现中作用一致,这里就不详细讲解。

三,hash迭代器

哈希表表支持的迭代器是单向迭代器,红黑树所支持的是双向迭代器。但是基础机构基本上相似,参考代码如下:

template<class K, class T, class Ptr, class Ref, class KeyOfT, class Hash> struct HTIterator { typedef HashNode<T> Node; typedef HTIterator<K, T, Ptr, Ref, KeyOfT, Hash> Self; Node* _node; const HashTable<K, T, KeyOfT, Hash>* _pht; HTIterator(Node* node, const HashTable<K, T, KeyOfT, Hash>* pht) :_node(node) , _pht(pht) {} };

由于迭代器要实现++操作。哈希表内部是链式结构,在迭代器++的过程中,当该节点所链接的节点结束,则需要寻找下一个数组的节点。为此不仅需要node指针指向节点,还需要哈希表指针指向该哈希表。

在该迭代器的模板的第二,三,四个参数分别代表这数据(数据类型)(数据指针类型)(数据引用类型)。方便后续区分const迭代器。在迭代器内部封装的结构和红黑树相似。

operator*

Ref operator*() { return _node->_data; }

operator->

Ptr operator->() { return &_node->_data; }

operator!=

bool operator!=(const Self& s) { return _node != s._node; }

operator++

Self& operator++() { if (_node->_next) { _node = _node->_next; } else { KeyOfT kot; Hash hs; size_t hashi = hs(kot(_node->_data)) % _pht > _tables.size(); ++hashi; while (hashi < _pht->_tables.size()) { if (_pht->_tables[hashi]) { break; } ++hashi; } if (hashi == _pht->_tables.size()) { _node = nullptr; // end() } else { _node = _pht->_tables[hashi]; } } return *this; }

若该节点的下一个位置还有节点,则直接node++。若已经到这条链的最后一个节点。则重新确定这个节点的位置。找到下一个非空节点。返回这个位置的迭代器。

四,HashTable的基本结构

template<class K, class T, class KeyOfT, class Hash> class HashTable { // 友元声明 template<class K, class T, class Ptr, class Ref, class KeyOfT, class Hash> friend struct HTIterator; typedef HashNode<T> Node; public: typedef HTIterator<K, T, T*, T&, KeyOfT, Hash> Iterator; typedef HTIterator<K, T, const T*, const T&, KeyOfT, Hash> private: vector<Node*> _tables; // size_t _n = 0; // };

这里需要注意类模板的友元声明。使得上文HTIterator能够访问HashTable的私有成员。

下面就是迭代器封装

Iterator Begin() { if (_n == 0) return End(); for (size_t i = 0; i < _tables.size(); i++) { Node* cur = _tables[i]; if (cur) { return Iterator(cur, this); } } return End(); } Iterator End() { return Iterator(nullptr, this); } ConstIterator Begin() const { if (_n == 0) return End(); for (size_t i = 0; i < _tables.size(); i++) { Node* cur = _tables[i]; if (cur) { return ConstIterator(cur, this); } } return End(); } ConstIterator End() const { return ConstIterator(nullptr, this); }

Insert的修改

Insert修改和这个基本上都类似。

五,my_unordered_set封装

基本结构和红黑树相似,与红黑树不同的是这里需要提供上文中所将到的KeyOFT,来返回key的值,参考代码如下:

struct SetKeyOfT { const K& operator()(const K& key) { return key; } };

基本结构如下:

template<class K, class Hash = HashFunc<K>> class unordered_set { struct SetKeyOfT { const K& operator()(const K& key) { return key; } }; public: typedef typename hash_bucket::HashTable<K, const K, SetKeyOfT, Hash>::Iterator iterator; typedef typename hash_bucket::HashTable<K, const K, SetKeyOfT, Hash>::ConstIterator const_iterator; private: hash_bucket::HashTable<K, const K, SetKeyOfT, Hash> _ht; };

这里需要注意不能单单用typedef,而是要使用typedef typename来强调后者修饰的是类型,而不是变量。map和set基本上一致。

http://www.rkmt.cn/news/89842.html

相关文章:

  • 一个卷积后就做池化还是多个卷积后做池化?
  • 智谱AI开源GLM-4-9B-Chat-1M:突破200万中文字符上下文壁垒,多模态能力引领行业新标杆
  • NCMconverter:解锁网易云音乐格式限制的终极解决方案
  • 知网AIGC检测原理是什么?知网AI率检测严格吗?
  • 论文降重与AIGC痕迹消除:当学术写作遇见宏智树AI学术
  • 液态智核V2震撼发布:重新定义边缘设备生成式AI体验
  • 斯坦福新框架AgentFlow突破AI决策瓶颈:模块化设计与Flow-GRPO训练法引领智能代理新范式
  • Kakao开源轻量级多模态模型Kanana-V:重新定义小参数视觉语言模型性能边界
  • Qwen3-235B-A22B-Instruct-2507震撼登场:256K超长上下文开启AI全场景应用新纪元
  • DeepSeek-Coder-V2-Instruct-0724强势登榜Aider LLM排行第二,技术突破引领代码大模型新高度
  • 18、Linux系统文件共享与安全防护指南
  • 21、Linux系统高级管理技巧全解析
  • 22、高级系统管理与故障排除技巧
  • Cesium快速入门16:Primitive多个实体与颜色修改
  • C语言实现堆排序(附带源码)
  • 后台任务与WebSocket实时应用
  • SQL分析函数`ROW_NUMBER`的兼容性与深度解析
  • Elasticsearch 的倒排索引原理
  • 一口气看懂 Android 操作系统架构 ——从“高层 App”一路挖到 “内核深处”
  • Kubernetes Master 节点核心组件全景解析
  • SolidWorks特征阵列类型及应用介绍
  • 2025年大语言模型生态全景:从技术突破到行业落地的多元发展态势
  • Python asyncio:解锁异步编程的魔法钥匙
  • 6
  • Trifucosyl(1-2,1-2,1-3)-iso-lacto-N-octaose—精准识别与靶向疗法的糖生物学关键工具 CAS:141342-93-0
  • 零延迟英雄锁定:League Akari智能选人系统深度解析
  • 深入解析Transformers 4.37:因果语言建模与掩码语言建模全流程实践指南
  • Z-image LoRA 训练整合包下载与使用教程(详细图文教程)
  • 神经网络中有超参数和自学习参数吗?
  • 突破AI推理天花板:GenSelect与TIR技术如何重塑大模型决策能力