C++:unordered_map与unordered_set详解

在这里插入图片描述

文章目录

  • 前言
  • 一、KeyOfT
    • 1. 为什么需要仿函数?
    • 2. MapKeyOfT与SetKeyOfT代码实现
  • 二、迭代器
    • 1. 设计背景
    • 2. 为什么需要存储哈希表指针
    • 3. operator++ 的逻辑
    • 4. begin() 和 end() 的实现
    • 5. 友元和前置声明的作用
    • 6. 完整代码
  • 三、迭代器map与set的复用
    • 1. map的复用,数据pair<K, V>
    • 2. Set的复用,数据Key
    • 3. 测试代码_1
  • 四、const迭代器——增加Ref与Ptr模板参数
  • 五、const迭代器权限放大的问题
  • 六、解决值不能修改的问题
  • 七、解决Insert返回值改为pair<iterator, bool>中Set的大坑
  • 八、operator[ ]
  • unordered_map与unordered_set封装代码总结


前言

参考源码框架,unordered_mapunordered_set 复用之前我们实现的哈希表。

  • 我们这里相比源码调整一下,key 参数就用 Kvalue 参数就用 V,哈希表中的数据类型,我们使用 T

  • 其次,跟 mapset 相比而言,unordered_mapunordered_set 的模拟实现类结构更复杂一点,但是大框架和思路是完全类似的。

    因为 HashTable 实现了泛型,不知道 T 参数导致是 K,还是 pair<K, V>,那么 insert 内部进行插入时要用 K 对象转换成整形取模和 K 比较相等。因为 pairvalue 不参与计算取模,且默认支持的是 keyvalue 一起比较相等,我们需要时的任何时候只需要比较 K 对象,所以我们在 unordered_mapunordered_set 层分别实现一个 MapKeyOfTSetKeyOfT 的仿函数传给 HashTableKeyOfT。然后 HashTable 中通过 KeyOfT 仿函数取出 T 类型对象中的 K 对象,再转换成整形取模和 K 比较相等。

具体细节参考如下代码实现。


一、KeyOfT

1. 为什么需要仿函数?

首先还是老问题,因为map与set存储的数据不同,泛型编程我们将数据改为T后对于map来说就是pair<K, V>,对于set来说就是key。

因此我们在插入的时候,比如要取出原来的kv.first全部都要变成kot(data)

在这里插入图片描述

这样用:
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述


2. MapKeyOfT与SetKeyOfT代码实现

#pragma once
#include"HashTable.h"

namespace jyf
{
	template<class K, class V>
	class unordered_map
	{
		struct MapKeyOfT
		{
			const K& operator()(const pair<K, V>& kv)
			{
				return kv.first;
			}
		};
	public:

	private:
		hash_bucket::HashTable<K, pair<K, V>, MapKeyOfT> _ht;
	};
}
#pragma once
#include"HashTable.h"

namespace jyf
{
	template<class K>
	class unordered_set
	{
		struct SetKeyOfT
		{
			const K& operator()(const K& key)
			{
				return key;
			}
		};
	public:

	private:
		hash_bucket::HashTable<K, K, SetKeyOfT> _ht;
	};
}

二、迭代器

接下来我们来实现迭代器:

1. 设计背景

哈希表的数据存储分散在多个桶(bucket)中,每个桶可能是一个链表,存储若干节点。因此,迭代器的实现不仅需要遍历链表中的节点,还需要跳跃到下一个非空桶继续遍历。为了实现这一功能,需要一个特殊的迭代器 HTIterator


2. 为什么需要存储哈希表指针

HTIterator 除了保存当前节点指针 _node,还需要保存一个指向哈希表对象的指针 _pht,主要原因是为了跨桶查找下一个节点:

template<class K, class T, class KeyOfT, class HashFunc>
struct HTIterator
{
	typedef HashNode<T> Node;
	typedef HTIterator<K, T, KeyOfT, HashFunc> Self;

	Node* _node;
	HashTable<K, T, KeyOfT, HashFunc>* _pht;

	HTIterator(Node* node, HashTable<K, T, KeyOfT, HashFunc>* pht)
		:_node(node)
		, _pht(pht)
	{}
  1. 桶定位需要哈希表结构:

    • 当遍历完当前桶后,operator++ 需要从当前桶的索引位置开始,查找下一个不为空的桶。
    • 这些桶存储在哈希表的 _table 成员中,迭代器需要通过 _pht 访问它。
  2. 简化设计:

    • 如果不存储哈希表指针,HTIterator 无法跨桶查找下一个节点,导致实现复杂且效率低下。
    • 存储 _pht 后,可以直接使用哈希表的结构信息(如桶的数量和内容)进行遍历。
  3. 提升代码灵活性:

    • _pht 提供了迭代器与哈希表的关联性,使得 HTIterator 能够独立于具体的哈希表实现,在其他类似结构中复用。

3. operator++ 的逻辑

operator++ 的核心任务是移动迭代器到下一个可访问的节点。分为两种情况:

  1. 当前桶的链表还有剩余节点:

    • 如果当前节点有 _next 指针,直接移动到链表中的下一个节点。
  2. 当前桶已经遍历完:

    • 获取当前桶索引: 根据当前节点的数据,通过哈希函数计算出当前桶的索引。
    • 查找下一个非空桶:
      • 从当前桶的下一个位置开始遍历 _table
      • 找到第一个不为空的桶后,将 _node 指向该桶的第一个节点。
      • 如果遍历到表尾仍未找到,将 _node 设置为 nullptr,表示迭代结束。

在这里插入图片描述

再次理解为什么要存储哈希表指针,如下图,我们需要表的数据记录hashi以及size。

在这里插入图片描述

代码实现:

Self& operator++()
{
	if (_node->_next)
	{
		// 当前桶还没完
		_node = _node->_next;
	}
	else
	{
		KeyOfT kot;
		HashFunc hf;
		size_t hashi = hf(kot(_node->_data)) % _pht->_table.size();
		// 从下一个位置查找查找下一个不为空的桶
		++hashi;
		while (hashi < _pht->_table.size())
		{
			if (_pht->_table[hashi])
			{
				_node = _pht->_table[hashi];
				return *this;
			}
			else
			{
				++hashi;
			}
		}

		_node = nullptr;
	}

	return *this;
}

4. begin() 和 end() 的实现

  • begin()

    • 从第一个桶开始查找,找到第一个非空桶,返回其第一个节点对应的迭代器。
    • 如果所有桶为空,返回空迭代器(iterator(nullptr, this))。
  • end()

    • 返回一个空迭代器(iterator(nullptr, this)),用于标志迭代结束。
      在这里插入图片描述

5. 友元和前置声明的作用

  1. 前置声明:
    • HTIterator 的定义在 HashTable 之后依赖于 HashTable 的定义,但 HashTable 的接口(如 begin()end())需要返回 HTIterator
    • 前置声明解决了这种相互依赖的问题,确保在定义 HashTable 时能够使用 HTIterator

在这里插入图片描述

  1. 友元声明:
    • HTIterator 需要访问 HashTable 的私有成员 _table_table.size(),这是其实现 operator++ 所必需的。
    • 通过友元声明,使 HTIterator 可以直接访问 HashTable 的私有成员,简化了代码设计。

在这里插入图片描述


6. 完整代码

#pragma once
#include

namespace hash_bucket
{
template
struct DefaultHashFunc
{
size_t operator() (const K& key)
{
return (size_t)key;
}
};

template<>
struct DefaultHashFunc<string>
{
	size_t operator() (const string& str)
	{
		// BKDR
		size_t hash = 0;
		for (auto ch : str)
		{
			hash *= 131;
			hash += ch;
		}

		return hash;
	}
};

template<class T>
struct HashNode
{
	T _data;
	HashNode<T>* _next;

	HashNode(const T& data)
		: _data(data)
		, _next(nullptr)
	{}
};

// 前置声明
template<class K, class T, class KeyOfT, class HashFunc>
class HashTable;

template<class K, class T, class KeyOfT, class HashFunc>
struct HTIterator
{
	typedef HashNode<T> Node;
	typedef HTIterator<K, T, KeyOfT, HashFunc> Self;

	Node* _node;
	HashTable<K, T, KeyOfT, HashFunc>* _pht;

	HTIterator(Node* node, HashTable<K, T, KeyOfT, HashFunc>* pht)
		:_node(node)
		, _pht(pht)
	{}

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

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

	Self& operator++()
	{
		if (_node->_next)
		{
			// 当前桶还没完
			_node = _node->_next;
		}
		else
		{
			KeyOfT kot;
			HashFunc hf;
			size_t hashi = hf(kot(_node->_data)) % _pht->_table.size();
			// 从下一个位置查找查找下一个不为空的桶
			++hashi;
			while (hashi < _pht->_table.size())
			{
				if (_pht->_table[hashi])
				{
					_node = _pht->_table[hashi];
					return *this;
				}
				else
				{
					++hashi;
				}
			}

			_node = nullptr;
		}

		return *this;
	}

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

template<class K, class T, class KeyOfT, class HashFunc = DefaultHashFunc<K>>
class HashTable
{
	typedef HashNode<T> Node;

	// 友元声明
	template<class K, class T, class KeyOfT, class HashFunc>
	friend struct HTIterator;
public:
	typedef HTIterator<K, T, KeyOfT, HashFunc> iterator;

	iterator begin()
	{
		// 找第一个桶
		for (size_t i = 0; i < _table.size(); i++)
		{
			Node* cur = _table[i];
			if (cur)
			{
				return iterator(cur, this);
			}
		}

		return iterator(nullptr, this);
	}

	iterator end()
	{
		return iterator(nullptr, this);
	}

	HashTable()
	{
		_table.resize(10, nullptr);
	}

	~HashTable()
	{
		for (size_t i = 0; i < _table.size(); i++)
		{
			Node* cur = _table[i];
			while (cur)
			{
				Node* next = cur->_next;
				delete cur;
				cur = next;
			}

			_table[i] = nullptr;
		}
	}

	bool Insert(const T& data)
	{
		// 仿函数控制
		KeyOfT kot;

		if (Find(kot(data)))
		{
			return false;
		}

		// 仿函数控制
		HashFunc hf;

		// 如果需要扩容
		if (_n == _table.size())
		{
			size_t newSize = _table.size() * 2;
			vector<Node*> newTable;
			newTable.resize(newSize, nullptr);

			// 遍历旧表,顺手牵羊,把节点牵下来挂到新表
			for (size_t i = 0; i < _table.size(); i++)
			{
				Node* cur = _table[i];
				while (cur)
				{
					Node* next = cur->_next;

					// 头插到新表
					size_t newhashi = hf(kot(cur->_data)) % newSize;
					cur->_next = newTable[newhashi];
					newTable[newhashi] = cur;

					cur = next;
				}

				_table[i] = nullptr;
			}

			_table.swap(newTable);
		}

		// 如果不需要扩容
		size_t hashi = hf(kot(data)) % _table.size();

		// 头插
		Node* newnode = new Node(data);
		newnode->_next = _table[hashi];
		_table[hashi] = newnode;

		++_n;
		return true;
	}

	Node* Find(const K& key)
	{
		HashFunc hf;
		KeyOfT kot;

		size_t hashi = hf(key) % _table.size();
		Node* cur = _table[hashi];

		while (cur)
		{
			if (kot(cur->_data) == key)
			{
				return cur;
			}

			cur = cur->_next;
		}

		return nullptr;
	}

	bool Erase(const K& key)
	{
		HashFunc hf;
		KeyOfT kot;

		size_t hashi = hf(key) % _table.size();
		Node* cur = _table[hashi];
		Node* prev = nullptr;

		while (cur)
		{
			if (kot(cur->_data) == key)
			{
				if (prev == nullptr)
				{
					_table[hashi] = cur->_next;
				}
				else
				{
					prev->_next = cur->_next;
				}

				delete cur;
				return true;
			}
			else
			{
				prev = cur;
				cur = cur->_next;
			}
		}

		return false;
	}

	void Print()
	{
		KeyOfT kot;

		for (size_t i = 0; i < _table.size(); i++)
		{
			printf("[%d]->", i);
			Node* cur = _table[i];
			while (cur)
			{
				cout << cur->_kv.first << ":" << cur->_kv.second << "->";
				cur = cur->_next;
			}
			printf("NULL\n");
		}
		cout << endl;

	}

private:
	vector<Node*> _table;     // 指针数组
	size_t _n = 0;            // 存储了多少个有效数据
};

}


三、迭代器map与set的复用

1. map的复用,数据pair<K, V>

#pragma once
#include"HashTable.h"

namespace jyf
{
	template<class K, class V>
	class unordered_map
	{
		struct MapKeyOfT
		{
			const K& operator()(const pair<K, V>& kv)
			{
				return kv.first;
			}
		};
	public:
		typedef typename hash_bucket::HashTable<K, pair<K, V>, MapKeyOfT>::iterator iterator;
		iterator begin()
		{
			return _ht.begin();
		}

		iterator end()
		{
			return _ht.end();
		}

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

	private:
		hash_bucket::HashTable<K, pair<K, V>, MapKeyOfT> _ht;
	};
}

2. Set的复用,数据Key

#pragma once
#include"HashTable.h"

namespace jyf
{
	template<class K>
	class unordered_set
	{
		struct SetKeyOfT
		{
			const K& operator()(const K& key)
			{
				return key;
			}
		};
	public:
		typedef typename hash_bucket::HashTable<K, K, SetKeyOfT>::iterator iterator;
		iterator begin()
		{
			return _ht.begin();
		}

		iterator end()
		{
			return _ht.end();
		}

		bool insert(const K& key)
		{
			return _ht.Insert(key);
		}
	private:
		hash_bucket::HashTable<K, K, SetKeyOfT> _ht;
	};
}

3. 测试代码_1

#include <iostream>
using namespace std;

#include"UnOrderedMap.h"
#include"UnOrderedSet.h"

int main()
{
	jyf::unordered_set<int> us;
	us.insert(3);
	us.insert(1);
	us.insert(3);
	us.insert(4);
	us.insert(5);
	us.insert(0);

	jyf::unordered_set<int>::iterator it = us.begin();
	while (it != us.end())
	{
		cout << *it << " ";
		++it;
	}
	cout << endl;

	jyf::unordered_map<string, string> dict;
	dict.insert(make_pair("sort", "排序"));
	dict.insert(make_pair("left", "左边"));
	dict.insert(make_pair("insert", "插入"));
	dict.insert(make_pair("sort", "xxx"));

	for (auto& kv : dict)
	{
		cout << kv.first << ":" << kv.second << endl;
	}

	return 0;
}

运行结果:
在这里插入图片描述


四、const迭代器——增加Ref与Ptr模板参数

const迭代器复用普通迭代器的代码,唯一不同的是operator*operator->的返回值,const迭代器返回的是const T*const T&,因此我们加入两个模板参数Ref与Ptr,通过typedef,我们实例化的时候就可以获得两种模式的迭代器:

在这里插入图片描述

在这里插入图片描述

这样子使用:
在这里插入图片描述

对于const迭代器来说:

const_iterator begin() const
{
	// 找第一个桶
	for (size_t i = 0; i < _table.size(); i++)
	{
		Node* cur = _table[i];
		if (cur)
		{
			return const_iterator(cur, this);
		}
	}

	return const_iterator(nullptr, this);
}

const_iterator end() const
{
	return const_iterator(nullptr, this);
}

五、const迭代器权限放大的问题

上述const迭代器还没有结束,const迭代器对于哈希这里还有一个坑!

在这里插入图片描述

那我们提供一个重载,用const HashTable<>*来接收可不可以呢?
在这里插入图片描述

因此我们需要把成员变量直接设置为const HashTable<>*
在这里插入图片描述
当然这样做了之后,我们就可以不用提供非const版本了~

在这里插入图片描述


六、解决值不能修改的问题

对于Set来说,key不可以修改,对于Map来说,pair<K,V>中的K不能修改,因此我们需要采取对应的措施。

对于Set来说:

普通迭代器与const迭代器都是const迭代器
在这里插入图片描述
在这里插入图片描述


对于Map来说:

Map的成员中的pair改为pair<const K, V>
迭代器正常走:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述


测试代码:

jyf::unordered_set<int> us;
us.insert(3);
us.insert(1);
us.insert(3);
us.insert(4);
us.insert(5);
us.insert(0);

jyf::unordered_set<int>::iterator it = us.begin();
while (it != us.end())
{
	// 不能修改key
	//*it += 10;
	cout << *it << " ";
	++it;
}
cout << endl;

jyf::unordered_map<string, string> dict;
dict.insert(make_pair("sort", "排序"));
dict.insert(make_pair("left", "左边"));
dict.insert(make_pair("insert", "插入"));
dict.insert(make_pair("sort", "xxx"));

jyf::unordered_map<string, string>::iterator dit = dict.begin();
while (dit != dict.end())
{
	// 不能修改key
	//dit->first += 'x';
	dit->second += 'x';

	cout << dit->first << ":" << dit->second << endl;


	++dit;
}
cout << endl;

七、解决Insert返回值改为pair<iterator, bool>中Set的大坑

接下来要做Insert返回值的修改,为重载operator[ ]做准备:

第一步,更改返回值:
在这里插入图片描述
在这里插入图片描述


第二步:find返回值也要对应修改
在这里插入图片描述
在这里插入图片描述


这里就会出现大问题:
现在跑不过了,是因为Set的大坑!
在这里插入图片描述


对于Map来说:
在这里插入图片描述


对于Set来说:
在这里插入图片描述

因此我们需要提供const_iterator的构造函数,使用iterator构造const_iterator.

在这里插入图片描述


八、operator[ ]

对于Map,我们还需要重载operator[],就像数组一样使用,可以插入,也可以查找,改Value。

这里我们要借助insert实现。

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

unordered_map与unordered_set封装代码总结

HashTable.h

#pragma once
#include<vector>
#include <string>


namespace hash_bucket
{
	template<class K>
	struct DefaultHashFunc
	{
		size_t operator() (const K& key)
		{
			return (size_t)key;
		}
	};

	template<>
	struct DefaultHashFunc<string>
	{
		size_t operator() (const string& str)
		{
			// BKDR
			size_t hash = 0;
			for (auto ch : str)
			{
				hash *= 131;
				hash += ch;
			}

			return hash;
		}
	};

	template<class T>
	struct HashNode
	{
		T _data;
		HashNode<T>* _next;

		HashNode(const T& data)
			: _data(data)
			, _next(nullptr)
		{}
	};

	// 前置声明
	template<class K, class T, class KeyOfT, class HashFunc>
	class HashTable;

	template<class K, class T, class Ref, class Ptr, class KeyOfT, class HashFunc>
	struct HTIterator
	{
		typedef HashNode<T> Node;
		typedef HTIterator<K, T, Ref, Ptr, KeyOfT, HashFunc> Self;

		typedef HTIterator<K, T, T&, T*, KeyOfT, HashFunc> iterator;

		Node* _node;
		const HashTable<K, T, KeyOfT, HashFunc>* _pht;

		/*HTIterator(Node* node, HashTable<K, T, KeyOfT, HashFunc>* pht)
			:_node(node)
			, _pht(pht)
		{}*/

		HTIterator(Node* node, const HashTable<K, T, KeyOfT, HashFunc>* pht)
			:_node(node)
			, _pht(pht)
		{}

		// 普通迭代器时,他是拷贝构造
		// const迭代器时,他是构造
		HTIterator(const iterator& it)
			: _node(it._node)
			, _pht(it._pht)
		{}

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

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

		Self& operator++()
		{
			if (_node->_next)
			{
				// 当前桶还没完
				_node = _node->_next;
			}
			else
			{
				KeyOfT kot;
				HashFunc hf;
				size_t hashi = hf(kot(_node->_data)) % _pht->_table.size();
				// 从下一个位置查找查找下一个不为空的桶
				++hashi;
				while (hashi < _pht->_table.size())
				{
					if (_pht->_table[hashi])
					{
						_node = _pht->_table[hashi];
						return *this;
					}
					else
					{
						++hashi;
					}
				}

				_node = nullptr;
			}

			return *this;
		}

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

	template<class K, class T, class KeyOfT, class HashFunc = DefaultHashFunc<K>>
	class HashTable
	{
		typedef HashNode<T> Node;

		// 友元声明
		template<class K, class T, class Ref, class Ptr, class KeyOfT, class HashFunc>
		friend struct HTIterator;
	public:
		typedef HTIterator<K, T, T&, T*, KeyOfT, HashFunc> iterator;
		typedef HTIterator<K, T, const T&, const T*, KeyOfT, HashFunc> const_iterator;

		iterator begin()
		{
			// 找第一个桶
			for (size_t i = 0; i < _table.size(); i++)
			{
				Node* cur = _table[i];
				if (cur)
				{
					return iterator(cur, this);
				}
			}

			return iterator(nullptr, this);
		}

		iterator end()
		{
			return iterator(nullptr, this);
		}

		const_iterator begin() const
		{
			// 找第一个桶
			for (size_t i = 0; i < _table.size(); i++)
			{
				Node* cur = _table[i];
				if (cur)
				{
					return const_iterator(cur, this);
				}
			}

			return const_iterator(nullptr, this);
		}

		const_iterator end() const
		{
			return const_iterator(nullptr, this);
		}

		HashTable()
		{
			_table.resize(10, nullptr);
		}

		~HashTable()
		{
			for (size_t i = 0; i < _table.size(); i++)
			{
				Node* cur = _table[i];
				while (cur)
				{
					Node* next = cur->_next;
					delete cur;
					cur = next;
				}

				_table[i] = nullptr;
			}
		}

		pair<iterator, bool> Insert(const T& data)
		{
			// 仿函数控制
			KeyOfT kot;

			iterator it = Find(kot(data));
			if (it != end())
			{
				return make_pair(it, false);
			}

			// 仿函数控制
			HashFunc hf;

			// 如果需要扩容
			if (_n == _table.size())
			{
				size_t newSize = _table.size() * 2;
				vector<Node*> newTable;
				newTable.resize(newSize, nullptr);

				// 遍历旧表,顺手牵羊,把节点牵下来挂到新表
				for (size_t i = 0; i < _table.size(); i++)
				{
					Node* cur = _table[i];
					while (cur)
					{
						Node* next = cur->_next;

						// 头插到新表
						size_t newhashi = hf(kot(cur->_data)) % newSize;
						cur->_next = newTable[newhashi];
						newTable[newhashi] = cur;

						cur = next;
					}

					_table[i] = nullptr;
				}

				_table.swap(newTable);
			}

			// 如果不需要扩容
			size_t hashi = hf(kot(data)) % _table.size();

			// 头插
			Node* newnode = new Node(data);
			newnode->_next = _table[hashi];
			_table[hashi] = newnode;

			++_n;
			return make_pair(iterator(newnode, this), true);
		}

		iterator Find(const K& key)
		{
			HashFunc hf;
			KeyOfT kot;

			size_t hashi = hf(key) % _table.size();
			Node* cur = _table[hashi];

			while (cur)
			{
				if (kot(cur->_data) == key)
				{
					return iterator(cur, this);
				}

				cur = cur->_next;
			}

			return end();
		}

		bool Erase(const K& key)
		{
			HashFunc hf;
			KeyOfT kot;

			size_t hashi = hf(key) % _table.size();
			Node* cur = _table[hashi];
			Node* prev = nullptr;

			while (cur)
			{
				if (kot(cur->_data) == key)
				{
					if (prev == nullptr)
					{
						_table[hashi] = cur->_next;
					}
					else
					{
						prev->_next = cur->_next;
					}

					delete cur;
					return true;
				}
				else
				{
					prev = cur;
					cur = cur->_next;
				}
			}

			return false;
		}

		void Print()
		{
			KeyOfT kot;

			for (size_t i = 0; i < _table.size(); i++)
			{
				printf("[%d]->", i);
				Node* cur = _table[i];
				while (cur)
				{
					cout << cur->_kv.first << ":" << cur->_kv.second << "->";
					cur = cur->_next;
				}
				printf("NULL\n");
			}
			cout << endl;

		}

	private:
		vector<Node*> _table;     // 指针数组
		size_t _n = 0;            // 存储了多少个有效数据
	};
}

UnOrderedMap.h

#pragma once
#include"HashTable.h"

namespace jyf
{
	template<class K, class V>
	class unordered_map
	{
		struct MapKeyOfT
		{
			const K& operator()(const pair<const K, V>& kv)
			{
				return kv.first;
			}
		};
	public:
		typedef typename hash_bucket::HashTable<K, pair<const K, V>, MapKeyOfT>::iterator iterator;
		typedef typename hash_bucket::HashTable<K, pair<const K, V>, MapKeyOfT>::const_iterator const_iterator;
		iterator begin()
		{
			return _ht.begin();
		}

		iterator end()
		{
			return _ht.end();
		}

		const_iterator begin() const
		{
			return _ht.begin();
		}

		const_iterator end() const
		{
			return _ht.end();
		}

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

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

	private:
		hash_bucket::HashTable<K, pair<const K, V>, MapKeyOfT> _ht;
	};
}

UnOrderedSet.h

#pragma once
#include"HashTable.h"

namespace jyf
{
	template<class K>
	class unordered_set
	{
		struct SetKeyOfT
		{
			const K& operator()(const K& key)
			{
				return key;
			}
		};
	public:
		typedef typename hash_bucket::HashTable<K, K, SetKeyOfT>::const_iterator iterator;
		typedef typename hash_bucket::HashTable<K, K, SetKeyOfT>::const_iterator const_iterator;

		const_iterator begin() const
		{
			return _ht.begin();
		}

		const_iterator end() const
		{
			return _ht.end();
		}

		pair<iterator, bool> insert(const K& key)
		{
			pair<typename hash_bucket::HashTable<K, K, SetKeyOfT>::iterator, bool> ret = _ht.Insert(key);
			return pair<iterator, bool>(ret.first, ret.second);
		}
	private:
		hash_bucket::HashTable<K, K, SetKeyOfT> _ht;
	};
}

test.cpp

#include <iostream>
using namespace std;

#include"UnOrderedSet.h"
#include"UnOrderedMap.h"

int main()
{
	jyf::unordered_set<int> us;
	us.insert(3);
	us.insert(1);
	us.insert(3);
	us.insert(4);
	us.insert(5);
	us.insert(0);

	jyf::unordered_set<int>::iterator it = us.begin();
	while (it != us.end())
	{
		// 不能修改key
		//*it += 10;
		cout << *it << " ";
		++it;
	}
	cout << endl;

	jyf::unordered_map<string, string> dict;
	dict.insert(make_pair("sort", "排序"));
	dict.insert(make_pair("left", "左边"));
	dict.insert(make_pair("insert", "插入"));
	dict.insert(make_pair("sort", "xxx"));

	jyf::unordered_map<string, string>::iterator dit = dict.begin();
	while (dit != dict.end())
	{
		// 不能修改key
		//dit->first += 'x';
		dit->second += 'x';

		cout << dit->first << ":" << dit->second << endl;


		++dit;
	}
	cout << endl;

	dict["sort"] = "排序";
	dict["insert"] = "插入";
	dict["string"] = "字符串";
	dict["right"];

	for (auto& kv : dict)
	{
		cout << kv.first << ":" << kv.second << endl;
	}

	return 0;
}

到这里就结束啦~
创作不易,如果对您有帮助的话,求一个一键三连,谢谢啦!

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

redis下载、基础数据类型、操作讲解说明,持久化、springboot整合等

1 Redis是什么 官网&#xff1a;https://redis.io 开发者&#xff1a;Antirez Redis诞生于2009年全称是Remote Dictionary Server 远程词典服务器&#xff0c;是一个基于内存的键值型NoSQL数据库。 Redis是一个开源的、高性能的键值对存储系统&#xff0c;它支持多种数据结构&…

《 C++ 修炼全景指南:二十五 》缓存系统的技术奥秘:LRU 原理、代码实现与未来趋势

摘要 本篇博客深入解析了 LRU&#xff08;Least Recently Used&#xff09;缓存机制&#xff0c;包括其核心原理、代码实现、优化策略和实际应用等方面。通过结合双向链表与哈希表&#xff0c;LRU 缓存实现了高效的数据插入、查找与删除操作。文章还对 LRU 的优化方案进行了详…

【k8s】给ServiceAccount 创建关联的 Secrets

说明 k8s v1.24.0 更新之后进行创建 ServiceAccount 不会自动生成 Secret 需要对其手动创建. 创建步骤 创建SA apiVersion: rbac.authorization.k8s.io/v1 kind: Role metadata:namespace: jtkjdevname: gitcicd-role rules: - apiGroups: ["apps"]resources: [&…

C++(4个类型转换)

1. C语言中的类型转换 1. 隐式 类型转换&#xff1a; 具有相近的类型才能进行互相转换&#xff0c;如&#xff1a;int,char,double都表示数值。 2. 强制类型转换&#xff1a;能隐式类型转换就能强制类型转换&#xff0c;隐式类型之间的转换类型强相关&#xff0c;强制类型转换…

Linux——命名管道及日志

linux——进程间通信及管道的应用场景-CSDN博客 文章目录 目录 文章目录 前言 一、命名管道是什么&#xff1f; 理解&#xff1a; 2、编写代码 makefile 管道封装成类&#xff0c;想用中管道时只需要调用实例化 读端 写端 日志 1、日志是什么&#xff1f; 2、日志有什么&#x…

动态代理如何加强安全性

在当今这个信息爆炸、网络无孔不入的时代&#xff0c;我们的每一次点击、每一次浏览都可能留下痕迹&#xff0c;成为潜在的安全隐患。如何在享受网络便利的同时&#xff0c;有效保护自己的隐私和信息安全&#xff0c;成为了每位网络使用者必须面对的重要课题。动态代理服务器&a…

5G学习笔记之随机接入

目录 1. 概述 2. MSG1 2.1 选择SSB 2.2 选择Preamble Index 2.3 选择发送Preamble的时频资源 2.4 确定RA-RNTI 2.5 确定发送功率 3. MSG2 4. MSG3 5. MSG4 6. 其它 6.1 切换中的随机接入 6.2 SI请求的随机接入 6.3 通过PDCCH order重新建立同步 1. 概述 随机接入…

《DSL-FIQA》论文翻译

《DSL-FIQA: Assessing Facial Image Quality Via Dual-Set Degradation Learning and Landmark-Guided Transformer》 原文链接&#xff1a;DSL-FIQA: Assessing Facial Image Quality via Dual-Set Degradation Learning and Landmark-Guided Transformer | IEEE Conference…

Redis实现限量优惠券的秒杀

核心&#xff1a;避免超卖问题&#xff0c;保证一人一单 业务逻辑 代码步骤分析 全部代码 Service public class VoucherOrderServiceImpl extends ServiceImpl<VoucherOrderMapper, VoucherOrder> implements IVoucherOrderService {Resourceprivate ISeckillVoucher…

STL算法之其它算法_中

目录 lower_bound(应用于有序区间) upper_bound&#xff08;应用于有序区间&#xff09; binary_search&#xff08;应用于有序区间&#xff09; next_permutation prev_permutation lower_bound(应用于有序区间) 这是二分查找(binary search)的一种版本&#xff0c;试图在…

Windows下从命令行(Powershell/CMD)发送内容到系统通知中心

Windows下从命令行&#xff08;Powershell/CMD&#xff09;发送内容到系统通知中心 01 前言 在平时写脚本的时候&#xff0c;将日志等信息直接输出到控制台固然是最直接的&#xff0c;而如果是一些后台执行的任务&#xff0c;不需要时刻关注运行细节但是又想知道一些大致的情…

计算机的错误计算(一百七十二)

摘要 探讨 MATLAB 对于算式 的计算误差。 例1. 在 MATLAB 中计算 的值。 直接贴图吧&#xff1a; 这样&#xff0c;MATLAB 的输出中只有3位正确数字&#xff0c;有效数字的错误率为 (16-3)/16 81.25% . 因为16位的正确输出为 0.2971242332737277e-18&#xff08;ISReals…

第30天:安全开发-JS 应用NodeJS 指南原型链污染Express 框架功能实现审计0

时间轴&#xff1a; 演示案例&#xff1a; 环境搭建-NodeJS-解析安装&库安装 功能实现-NodeJS-数据库&文件&执行 安全问题-NodeJS-注入&RCE&原型链 案例分析-NodeJS-CTF 题目&源码审计 开发指南-NodeJS-安全 SecGuide 项目、 环境搭建-NodeJ…

SQL优化与性能——数据库事务管理

数据库事务管理是数据库系统中至关重要的一部分&#xff0c;确保了数据的一致性、完整性、可靠性和隔离性。尤其在高并发、高负载的系统中&#xff0c;事务管理的设计和实现直接影响到系统的稳定性和性能。本章将详细探讨以下内容&#xff1a;事务的ACID特性、使用 BEGIN、COMM…

Rook入门:打造云原生Ceph存储的全面学习路径(上)

文章目录 一.Rook简介二.Rook与Ceph架构2.1 Rook结构体系2.2 Rook包含组件2.3 Rook与kubernetes结合的架构图如下2.4 ceph特点2.5 ceph架构2.6 ceph组件 三.Rook部署Ceph集群3.1 部署条件3.2 获取rook最新版本3.3 rook资源文件目录结构3.4 部署Rook/CRD/Ceph集群3.5 查看rook部…

机器学习——生成对抗网络(GANs):原理、进展与应用前景分析

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一. 生成对抗网络的基本原理二. 使用步骤2.1 对抗性训练2.2 损失函数 三. GAN的变种和进展四. 生成对抗网络的应用五. 持续挑战与未来发展方向六. 小结 前言 生…

IDEA连接Apifox客户端

IDEA连接Apifox客户端 一、下载Apifox安装包二、IDEA配置三、配置Apifox和IDEA项目同步 一、下载Apifox安装包 Apifox官网&#xff0c;根据自己的操作系统下载对应的Apifox安装包&#xff0c;我是windows系统所以下载的是windows版。 下载 默认仅为我安装&#xff0c;点击下一…

Python毕业设计选题:基于django+vue的校园影院售票系统

开发语言&#xff1a;Python框架&#xff1a;djangoPython版本&#xff1a;python3.7.7数据库&#xff1a;mysql 5.7数据库工具&#xff1a;Navicat11开发软件&#xff1a;PyCharm 系统展示 管理员登录 管理员功能界面 用户管理 影院信息管理 电影类型管理 电影信息管理 系统…

《Java核心技术I》线程状态

12.2 线程状态 线程可以有6种状态&#xff1a; New(新建)Runnable(可运行)Blocked(阻塞)Waiting(等待)Timed waiting(计时等待)Terminated(终止) 确定当前线程的状态&#xff0c;只需要调用getState()方法。 12.2.1 新建线程 当new创建一个线程时&#xff0c;线程还未运行…

树莓派基本配置-基础配置配置

树莓派基本配置 文章目录 树莓派基本配置前言硬件准备树莓派刷机串口方式登录树莓派接入网络ssh方式登录树莓派更换国内源xrdp界面登录树莓派远程文件传输FileZilla 前言 树莓派是一款功能强大且价格实惠的小型计算机&#xff0c;非常适合作为学习编程、物联网项目、家庭自动化…