【数据结构】【C++】封装哈希表模拟实现unordered_map和unordered_set容器

news2025/1/18 6:50:34

【数据结构】&&【C++】封装哈希表模拟实现unordered_map和unordered_set容器

  • 一.哈希表的完成
  • 二.改造哈希表(泛型适配)
  • 三.封装unordered_map和unordered_set的接口
  • 四.实现哈希表迭代器(泛型适配)
  • 五.封装unordered_map和unordered_set的迭代器
  • 六.解决key不能修改问题
  • 七.实现map[]运算符重载

一.哈希表的完成

在上一篇哈希表的模拟实现中,已经将哈希表实现完毕,这里不再分析。

#pragma once
using namespace std;
#include <vector>
#include<iostream>



//哈希桶

//1.第一哈希表的完成

//哈希结点
template <class K,class V>
struct HashNode
{
	pair<K, V> _kv;
	//存储的数据
	HashNode<K, V>* _next;
	//指向下一个结点的指针

	HashNode(const pair<K,V> kv)
		:_kv(kv)
		,_next(nullptr)
	{}
};

template <class K>
struct defaulthashfunc//默认的仿函数可以让数据模
{
	size_t operator()(const K& key)
	{
		return (size_t)key;
		//将key强行类型转化
		//这样作的意义:可以负数也可以进行模了
	}
};
template <>
//模板的特化,当数据类型是int的时候就默认使用defaulthashfunc<int>,当数据类型是string类型时,就默认使用defaulthashfunc<string>
struct defaulthashfunc<string>
{
	size_t operator()(const string& str)
	{
		//为了减少冲突,我们将字符串的每个字符的值相加
		size_t hash = 0;
		for (auto& it : str)
		{
			hash *= 131;
			hash += it;
		}
		return hash;
	}
};

//要写一个仿函数?  因为不是所有的数据类型都可以模的
// 一般整数是可以模的,string类型是无法模的
// 所以我们要写一个仿函数来达到传的数据可以模
//这样也就是增加了哈希表的一个模板参数l
template<class K,class V,class Hashfunc=defaulthashfunc<K>>
//哈希表
class Hash_table
{
	typedef HashNode<K, V> Node;


	//哈希需要将写析构函数,虽然自定义类型vector会自动调用默认析构,但它里面的成员是内置类型,没有默认构造,
	//所以需要我们自己析构每个结点
public:
	~Hash_table()
	{
		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;
		}
	}
	Hash_table()
	{
		//构造
		_table.resize(10, nullptr);
		//首先开出十个空间,每个空间值为空
	}

	bool insert(const pair<K, V> _kv)
	{

		 Hashfunc hf;
		//仿函数可以使数据模
		//在插入之前,确认一下表里是否已经有了这个值,如果有了就不用插入了
		if (find(_kv.first))
		{
			return false;
		}

		//我们自己控制扩容逻辑,虽然vector会自己扩容,但我们要控制。因为扩容完,有的key会冲突有的值又不冲突了。
	   //如果不扩容,那么冲突多了就根单链表一样了
		//当负载因子大约等于1时就要扩容,平均每个桶里有一个数据

		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++)
			{
				//扩容后,空间size变大了,有的数据就可能会存到不同的桶里了
				//拿下来的结点要重新计算放进哪个位置
				Node* cur = _table[i];
				//cur后面可能还有链接的结点
				while (cur)
				{
					size_t hashi = hf(cur->_kv.first) % newtable.size();
					

					Node* next = cur->_next;
					//头插到新表
					cur->_next = newtable[hashi];
					//头插 这个结点的 接到插入结点的前面对吧
					//那么next就接到newtavle[i]
					newtable[hashi] = cur;

					//往后走接着拿走
					cur = next;
				}

				//当前桶里的结点被拿光后,就置为空
				_table[i] = nullptr;


			}
			//这个新表就是我们想要的表,那么我们利用vector、的交换,让旧表和新表交换

			_table.swap(newtable);
		}

		size_t hashi = hf(_kv.first) % _table.size();

		Node* newnode = new Node(_kv);
		newnode->_next = _table[hashi];
		_table[hashi] = newnode;
		//将新结点头插到哈希桶里
		++n;
		return true;
	}
	Node* find(const K& key)
	{

		Hashfunc hf;
		size_t hashi = hf(key)% _table.size();

		Node* cur = _table[hashi];
		
		while (cur)
		{
			if (cur->_kv.first == key)
				return cur;
			else
				cur = cur->_next;
		}
		return nullptr;
	}
	void Print()
	{
		for (int i = 0; i < _table.size(); i++)
		{
			printf("[%d]", i);

			Node* cur = _table[i];
			while (cur)
			{
				cout << cur->_kv.first << " ";
				cur = cur->_next;
			}
			cout << endl;
			}

	}
	bool erase(const K& key)
	{

		Hashfunc hf;
		//可以复用find吗?先用find找到key然后删除key呢?4
		//不可以,因为删除一个结点需要找到这个结点的前面和后面的位置,但这里只有key的位置,所以不能直接复用find,但是复用其中的步骤
		size_t hashi = hf(key) % _table.size();

		Node* cur = _table[hashi];
		Node* prev = nullptr;
		while (cur)
		{
			if (cur->_kv.first == key)//找到要删除的结点后
			{
				
				//将前面的结点的指针指向后面的前面

				//还有一种可能cur就是桶里的第一个,那么就是头删了,prev就是nullptr
				if (prev == nullptr)
				{
					_table[hashi] = cur->_next;
				}
				else
				{
					prev->_next = cur->_next;
				}
			
				delete cur;
				return true;
			}
			else
			{
				prev = cur;
				//每次先记录一下前面的结点位置
				cur = cur->_next;
			}
				
		}
		return false;

	}
private:
	//底层封装的是一个数组
	//数组里的指针指向的都是哈希结点
	vector<Node*> _table;
	//底层还封装一个大小n表示实际表中的数据个数
	size_t n=0;///用来计算负载因子
};

二.改造哈希表(泛型适配)

我们要对哈希表进行改造,因为unordered_map和unordered_set底层用的都是哈希表,虽然不是同一个哈希表,但是是同一个模板实例化出来的哈希表。我们要让哈希表可以适配存储不同的数据类型,因为unordered_set里面存的是K类型,而unordered_map里面存的是pair<K,V>类型。
所以我们一开始并不知道哈希表里存的数据是什么类型,那么就用T表示。当传的模板参数是K类型,哈希表就实例化存的就是K类型,当传的模板参数是pair<K,V>类型,哈希表实例化存的就是pair<K,V>类型,所以我们可以通过传不同的模板参数来决定哈希表里存的是什么数据类型。
所以我们需要改造哈希表,将里面的数据都改成T类型,修改成T类型的原因是我们不知道是什么数据类型,根据传的模板参数决定。


template <class T>
//定义哈希结点
struct HashNode
{
	T _data;
	//存储的数据是T类型,根据传过来的模板参数确定
	HashNode<T>* _next;
	//指向下一个结点的指针

	HashNode(const T& data)
		:_data(data)
		, _next(nullptr)
	{}
};
// [问题]:一旦泛型后,我们就不知道数据的具体类型了,那么我们要利用除留余数法计算哈希地址时(我们都是利用key来进行取模的而我们不知道具体的类型是K类型还是pair<K,V>类型),就要写一个仿函数,这个仿函数根据map和set传过来的,然后实例化出哈希表中的仿函数。获取数据中的key,让key进行计算

```c
template<class K, class T>
//哈希表
class Hash_table
{
	typedef HashNode<T> Node;

public:
	
	~Hash_table()
	{
		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;
		}
	}
	Hash_table()
	{
		//构造
		_table.resize(10, nullptr);
		//首先开出十个空间,每个空间值为空
	}

	bool insert(const T& data)
	{

	 …………………
	}
	bool find(const K& key)
	{

	 ………………
	}
	
	bool erase(const K& key)
	{

	 ………………	
	}
private:
	vector<Node*> _table;
	//底层还封装一个大小n表示实际表中的数据个数
	size_t n = 0;//用来计算负载因子
};

存在问题:
这里不管是插入还是查找还是删除第一步都是需要将数据的哈希地址找到,而哈希地址是利用除留余数法计算得到的,是利用key值进行取模的,但这里一旦适配泛型后,我们就不知道具体的类型是K类型还是pair<K,V>类型,如果是K类型那么就可以直接取模,如果是pair<K,V>类型,那是不可以直接取模的,需要将里面的key值取出来。
解决方法:
我们可以利用一个仿函数,这个仿函数的功能是可以将数据里的Key类型数据取出来。那么我们可以给哈希表增加一个模板参数,给仿函数用。一旦遇到要计算哈希地址或者比较的操作时,我们就可以将数据里的K值取出来进行计算比较。
仿函数实现的原理:当T类型是K类型数据时,直接返回K值即可,当T类型是pair<K,V>数据时,返回里面的first数据即可(就是K值)。

template<class K, class T, class KeyOfT,class Hashfunc = defaulthashfunc<K>>
//哈希表
class Hash_table
{
	typedef HashNode<T> Node;
public:
	Hash_table()
	{
		//构造
		_table.resize(10, nullptr);
		//首先开出十个空间,每个空间值为空
	}

	bool insert(const T& data)
	{

		KeyOfT kt;
		Hashfunc hf;
		//仿函数可以使数据模
		//在插入之前,确认一下表里是否已经有了这个值,如果有了就不用插入了
		
		if (find(kt(data)))
		{
			return false;
		}

		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];
				//cur后面可能还有链接的结点
				while (cur)
				{
					size_t hashi = hf(kt(cur->_data)) % newtable.size();
					Node* next = cur->_next;
					cur->_next = newtable[hashi];
					newtable[hashi] = cur;

					//往后走接着拿走
					cur = next;
				}

				//当前桶里的结点被拿光后,就置为空
				_table[i] = nullptr;


			}
			//这个新表就是我们想要的表,那么我们利用vector、的交换,让旧表和新表交换

			_table.swap(newtable);
		}

		size_t hashi = hf(kt(data)) % _table.size();
		//这里有两个仿函数kt是用来获取data数据里的key值
		//hf是用来适配不同的K值,因为string类型无法取模
		
		Node* newnode = new Node(data);
		newnode->_next = _table[hashi];
		_table[hashi] = newnode;
		//将新结点头插到哈希桶里
		++n;
		return true;
	}
	iterator find(const K& key)
	{

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

		while (cur)
		{
			if (kt(cur->_data) == key)
				return iterator(cur,this);
			else
				cur = cur->_next;
		}
		return iterator(nullptr,this);
	}
	bool erase(const K& key)
	{

		Hashfunc hf;
		KeyOfT kt;
		size_t hashi = hf(key) % _table.size();

		Node* cur = _table[hashi];
		Node* prev = nullptr;
		while (cur)
		{
		//这里仿函数kt是用来获取data数据里的key值
			if (kt(cur->_data) == key)//找到要删除的结点后
			{

				//将前面的结点的指针指向后面的前面

				//还有一种可能cur就是桶里的第一个,那么就是头删了,prev就是nullptr
				if (prev == nullptr)
				{
					_table[hashi] = cur->_next;
				}
				else
				{
					prev->_next = cur->_next;
				}

				delete cur;
				--n;
				return true;
			}
			else
			{
				prev = cur;
				//每次先记录一下前面的结点位置
				cur = cur->_next;
			}

		}
		return false;
	}
private:
	vector<Node*> _table;
	//底层还封装一个大小n表示实际表中的数据个数
	size_t n = 0;//用来计算负载因子
};

三.封装unordered_map和unordered_set的接口

封装set,内部实现仿函数,然后底层封装的是存储K值(结点指针)的哈希表。

#include"Hash.h"
namespace tao
{
	template<class K>
	
	class set
	{
		struct setoft//仿函数用来获取数据的key值
		{
			const K& operator()(const K& key)
			{
				return key;
			}
		};
		
	public:
		
		bool insert(const K& key)
		{
			return _ht.insert(key);
		}
	private:
		//底层封装一个哈希表,哈希表里存的是K类型
		Hash_table<K, K, setoft> _ht;
	};
};

封装map,实现仿函数,底层存储的是pair<K,V>类型的(结点指针)哈希表。

#include"Hash.h"
namespace tao
{

	template<class K, class V>
	
	class map
	{
		struct mapoft//仿函数获取数据中的key值
		{
			const K& operator()(const pair<K, V>& kv)
			{
				return kv.first;
			}
		};

	public:

		bool insert(const pair<K, V>& kv)
		{
			return _ht.insert(kv);
		}
	private:
		//底层封装一个哈希表,哈希表里的数据是pair<K,V>类型
		Hash_table<K, pair<const K, V>, mapoft> _ht;
	};
};

四.实现哈希表迭代器(泛型适配)

哈希表的迭代器是一个自定义类型,因为原生的结点指针不能满足我们想要的操作,所以我们就直接将结点的指针封装起来,然后自定义一个类型,对这个结点指针增加一些操作来完成我们想要的效果。
首先哈希表的迭代器里肯定要封装一个结点的指针,++后找到下一个结点,在一个哈希桶里,++就代表着找链表下面的结点即可,那么如果该结点就是链表最后一个结点呢?怎么找到下一个不为空的桶呢?又或者你是如何找到第一个不为空的桶的呢?
所以这里我们还需要一个指向哈希表的指针,这样我们才可以找到桶与桶之间的关系,而结点指针是用来找结点与结点之间的关系的。

哈希表的迭代器里封装两个对象,一个是结点指针,一个哈希表指针。

1.迭代器有普通迭代器和const迭代器,普通迭代很好实现,那么const迭代器如何实现呢?
与链表的const迭代器实现原理一样,我们通过三个模板参数(template <class T,class Ref,class Ptr>)来控制函数的返回值,从而控制返回的是普通类型的迭代器还是const类型的迭代器。这里也就是泛型适配,适配多种类型 。

Ref控制解引用函数的返回值,当Ref为T&时,返回的就是普通迭代器,当Ref为const T&时,返回的就是const迭代器。
Ptr控制的->重载函数的返回值,当Ptr为T时,返回的就是普通迭代器,当Ptr为const T时,返回的就是const迭代器。

2.哈希表迭代器的++实现原理:

①假设当前结点在一个桶里,这个桶还没走完,那么直接找下一个结点即可
②如果这个结点是桶里最后一个原生,即桶走完了,那么我们就要找下一个不为空的桶
③如何找到下一个不为空的桶呢?首先将当前结点的哈希地址计算出来,然后将哈希地址++,再利用一个循环,查找后面的桶是否为空,如果不为空,那么这个桶就是最终结果,如果为空就再找后面的桶。

//泛型适配--适配普通迭代器和const迭代器
template<class K, class T,class Ref,class Ptr, class KeyOfT, class Hashfunc>
//哈希表的迭代器里面肯定封装一个结点的指针,但还需要通过表来找到下一个桶,因为我们需要遍历先找到第一个桶,当这个桶遍历完后,怎么找到下一个不为空的桶呢?
//需要通过这个哈希表来找到桶,所以我们还需要一个指向哈希表的指针

struct HSIterator
{
	
	typedef  HashNode<T> Node;
	/
	typedef HSIterator<K, T,Ref,Ptr,KeyOfT, Hashfunc> Self;
	Node* _node;
	//底层封装着一个结点指针
	const Hash_table <K,T, KeyOfT,Hashfunc>* _pht;
	//底层还封装着一个哈希表指针
	//这里可以加const因为我们不是根据pht来找到哈希表来修改哈希表里的内容

	
	HSIterator(Node* node, Hash_table<K, T, KeyOfT,Hashfunc>* hs)
		:_node(node)
		, _pht(hs)
	{}
	
	Ref& operator*()
	{
		return _node->_data;
	}
	Ptr operator->()
	{
		return &_node->_data;
	}
	Self& operator++()
	{
		if (_node->_next)//当前桶没有走完,那么直接找下一个即可
		{
			_node = _node->_next;
		}
		else//当前桶走到空了,就要找下一个不为空的桶
		{
			KeyOfT kof;
			Hashfunc hf;

			size_t hashi = hf(kof(_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;
	}
	
};

【存在问题1】:
这里存在一个相互依赖关系问题,因为在哈希表里我们使用了迭代器,在迭代器里我们又使用了哈希表。我们需要在这里使用前置声明告诉编译器,我们是有哈希表,只不过哈希表在后面。这样就不会报错啦。

//这里存在问题,迭代器里要用哈希,哈希里面要有迭代器,相互依赖关系
//我们这里用前置声明告诉编译器,我们要是有哈希表,这个表是存在的,在后面
template<class K, class T, class KeyOfT, class Hashfunc>
class Hash_table;
template<class K, class T,class Ref,class Ptr, class KeyOfT, class Hashfunc>

struct HSIterator
{
   ………………………………………………………………
   ………………………………………………………………
};

【存在问题2】:
在迭代器的++里我们在计算当前结点的哈希地址时,取模时,利用哈希指针找到了哈希表里的vector<Node*> _table元素,并访问了它的函数,这里我们在外面调用哈希表的私有成员,这样是不可行,所以我们需要让迭代器成为哈希表的友元类,这样在迭代器里就可以使用哈希表的私有成员了。
在这里插入图片描述

迭代器实现之后,我们就可以在哈希表里,来实现迭代器的begin()和end()了。
begin()就是找哈希表里第一个不为空的桶。
end()就是找最后一个不为空的桶的下一个位置,也就是空。

template<class K, class T, class KeyOfT,class Hashfunc = defaulthashfunc<K>>
//哈希表
class Hash_table
{
	typedef HashNode<T> Node;

	template<class K, class T, class Ref,class Ptr,class KeyOfT, class Hashfunc>
	friend struct HSIterator;
	//让迭代器成为哈希表的友元
public:
	typedef HSIterator<K, T,T&,T*, KeyOfT, Hashfunc> iterator;
	//适配普通迭代器
	typedef HSIterator<K, T, const T&, const T*, KeyOfT, Hashfunc> const_iterator;
	//适配const迭代器
	iterator begin()
	{
		//找第一个桶
		for (size_t i = 0; i < _table.size(); i++)
		{
			Node* cur = _table[i];
			while (cur)
			{
				if (cur)
				{
					return iterator(cur, this);
					//这里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];
			while (cur)
			{
				if (cur)
				{
					return const_iterator(cur, this);
				}
			}
		}
		//最后没有找到
		return const_iterator(nullptr, this);
	}
	
	const_iterator end()const
	{
		return const_iterator(nullptr, this);
	}


	bool insert(const T& data)
	{
      …………………………
	}
	bool erase(const K& key)
	{
      …………………………
	}
private:
	vector<Node*> _table;
	//底层还封装一个大小n表示实际表中的数据个数
	size_t n = 0;//用来计算负载因子
};

这样哈希表的迭代器就完成啦。不过这里还有一个问题喔,确实封装哈希表确实比较恶心,比封装红黑树还复杂。
这里的问题在于const修饰begin()和const修饰end()这里是编译不过的。为什么呢?这里提醒没有构造函数可以接收,或者构造函数重载不明确。
在这里插入图片描述
这里的原因是因为const修饰了this指针,导致指向哈希表的指针变成const类型了,而迭代器的构造里,是用普通迭代器构造的。所以当this指针传过去构造时,const是不能传给普通类型的,权限放大了。所以这里我们只需要重载一个参数类型是const类型的哈希表指针即可。

	HSIterator(Node* node, Hash_table<K, T, KeyOfT,Hashfunc>* hs)//hs是普通的类型
		:_node(node)
		, _pht(hs)
	{}
	//当传过来的哈希表指针是普通指针就走上面--权限的缩小
	//重载一个,当传过来的是const修饰的哈希表指针就走这里---权限的平移
	HSIterator(Node* node, const Hash_table<K, T, KeyOfT, Hashfunc>* hs)//hs是普通的类型
		:_node(node)
		, _pht(hs)
	{}

五.封装unordered_map和unordered_set的迭代器

只有哈希表里的迭代器完成了,才可以封装map和set里的迭代器。
封装set的迭代器,本质就是调用哈希表的迭代器接口。

1.不过要注意的是,在重命名红黑树里的迭代器时,需要在类名前面加上typename,如果不加上typename是不行的,因为这时类模板还没有实例化出对象出来,就算实例化了,也有部分类型没有实例,因为编译器也不知道这个是内置类型还是静态变量,加上是告诉编译器这个是类型,这个类型在类模板里定义,等类模板实例化后再找。
2.定义好普通迭代和const迭代器后,就可以实现begin()和end()了。

namespace tao
{
	template<class K>
	class set
	{
		struct setoft
		{
			const K& operator()(const K& key)
			{
				return key;
			}
		};
		
	public:
		typedef typename Hash_table<K, K, setoft>::iterator iterator;
		typedef typename Hash_table<K, K, setoft>::const_iterator const_iterator;
		
		iterator begin()
		{
			return _ht.begin();
		}
		iterator end()
		{
			return _ht.end();
		}
		
		iterator begin()const
		{
			return _ht.begin();
		}
		iterator end()const
		{
			return _ht.end();
		}
		
		bool insert(const K& key)
		{
			return _ht.insert(key);
		}
	private:
		//底层封装一个哈希表,哈希表里存的是K类型
		Hash_table<K, K, setoft> _ht;
	};
};


封装map的迭代器,本质上就是调用哈希表里的迭代器接口。

#include"Hash.h"
namespace tao
{
	template<class K, class V>
	class map
	{
		struct mapoft
		{
			const K& operator()(const pair<K, V>& kv)
			{
				return kv.first;
			}
		};
		
	public:
		typedef typename Hash_table<K, pair<K, V>, mapoft>::iterator iterator;
		typedef typename Hash_table<K, pair< K, V>, mapoft>::const_iterator const_iterator;
		iterator begin()
		{
			return _ht.begin();
		}
		iterator end()
		{
			return _ht.end();
		}
		iterator begin()const
		{
			return _ht.begin();
		}
		iterator end()const
		{
			return _ht.end();
		}

		bool insert(const pair<K, V>& kv)
		{
			return _ht.insert(kv);
		}
	private:
		//底层封装一个哈希表,哈希表里的数据是pair<K,V>类型
		Hash_table<K, pair< K, V>, mapoft> _ht;
	};
};

六.解决key不能修改问题

set里面存储的Key值是不能被修改的,map里的存储的K值也是不能被修改,但是Value值是可以被修改!
如果解决这个问题呢?

问题:set里的key值不能被修改。map里的key值不能被修改,value值可以被修改。 set解决原理:
1.set里存储的值就只有Key值,索性我们直接让这个存储的数据无法被修改,只能访问读取,无法修改。即使用const修饰。而我们是通过迭代器来访问到这个数据的,所以我们让普通迭代器变成const迭代器即可。所以在set里,普通迭代器和const迭代器最终都是const迭代器。
2.那么迭代器都是const的了,最终都只会调用const修饰的begin()和end()函数了,普通的begin()和end()就不需要写了。
3.不过这样处理又会出现一个很难搞的问题,这个就是set的insert的返回值问题,我们后面要实现map的[]运算符重载就会讲到。

set的解决方法:

	public:
		typedef typename Hash_table<K, K, setoft>::const_iterator iterator;
		//为了解决set的key不能被修改所以我们让普通迭代器变成const迭代器
		typedef typename Hash_table<K, K, setoft>::const_iterator const_iterator;
		//因为普通迭代器也是const所以我们只用写const的begin和end即可。
		iterator begin()const
		{
			return _ht.begin();
		}
		iterator end()const
		{
			return _ht.end();
		}
		//这里的pair<const_iterator,bool>类型的
		bool insert(const K& key)
		{
			//return _ht.insert(key);
		}
	private:
		//底层封装一个哈希表,哈希表里存的是K类型
		Hash_table<K, K, setoft> _ht;
	};
};


map的解决原理
1.在存储的时候就让K值无法修改。
2.因为我们知道map里存储的数据是pair<K,V>类型,我们不能想set那个让普通迭代器变成const迭代器,因为map要求Value的值还是可以修改的,所以不让pair<K,V>类型无法修改,而是单纯的让里面的K值无法修改,也就是在里面用const修饰K,那么这样K值就不能被修改,V值可以被修改。
3.pair是可以修改的,但是里面的K是无法被修改的!

map的解决方法:
	public:
		typedef typename Hash_table<K, pair<const K, V>, mapoft>::iterator iterator;
		typedef typename Hash_table<K, pair<const K, V>, mapoft>::const_iterator const_iterator;
		iterator begin()
		{
			return _ht.begin();
		}
		iterator end()
		{
			return _ht.end();
		}
		iterator begin()const
		{
			return _ht.begin();
		}
		iterator end()const
		{
			return _ht.end();
		}
		
		bool insert(const pair<K, V>& kv)
		{
			return _ht.insert(kv);
		}
	private:
		//底层封装一个哈希表,哈希表里的数据是pair<K,V>类型
		Hash_table<K, pair<const K, V>, mapoft> _ht;
	};
};

七.实现map[]运算符重载

map的[ ]运算符重载,底层实现本质是调用了insert函数。然后通过insert函数返回的pair<iterator,bool>类型数据来找到Value值。

所以在实现[ ]运算符重载时,我们需要对哈希表里的insert进行改造,因为原来的insert的返回值是布尔值,我们需要pair类型返回值。

pair<iterator,bool> insert(const T& data)
	{

		KeyOfT kt;
		Hashfunc hf;
		//仿函数可以使数据模
		//在插入之前,确认一下表里是否已经有了这个值,如果有了就不用插入了
		iterator it = find(kt(data));
		if (it!=end())
		{
			return make_pair(it,false);
		}
		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++)
			{
				//扩容后,空间size变大了,有的数据就可能会存到不同的桶里了
				//拿下来的结点要重新计算放进哪个位置
				Node* cur = _table[i];
				//cur后面可能还有链接的结点
				while (cur)
				{
					size_t hashi = hf(kt(cur->_data)) % newtable.size();


					Node* next = cur->_next;
					//头插到新表
					cur->_next = newtable[hashi];
					//头插 这个结点的 接到插入结点的前面对吧
					//那么next就接到newtavle[i]
					newtable[hashi] = cur;

					//往后走接着拿走
					cur = next;
				}

				//当前桶里的结点被拿光后,就置为空
				_table[i] = nullptr;


			}
			//这个新表就是我们想要的表,那么我们利用vector、的交换,让旧表和新表交换

			_table.swap(newtable);
		}

		size_t hashi = hf(kt(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 kt;
		size_t hashi = hf(key) % _table.size();

		Node* cur = _table[hashi];

		while (cur)
		{
			if (kt(cur->_data) == key)
				return iterator(cur,this);
			else
				cur = cur->_next;
		}
		return iterator(nullptr,this);
	}

哈希表的insert改造后,那么set和map里的insert都需要修改,因为底层用的就是调用用哈希表的insert。

在这里插入图片描述
这样修改是对的吗?有没有什么问题呢?
但是这时会出现一个问题,set里面的insert报错。这是为什么呢?
在这里插入图片描述
问题在于,我们之前让普通迭代变成const迭代器,而这里的pair<iterator,bool>中的iterator其实本质上是const_iterator。
是pair<const_itearto,bool>类型的。而哈希表里的insert返回的是普通迭代器,也就是pair<iterator,bool>类型的。这是两个不同的类型,无法直接将pair<iterator,bool>类型转换成pair<const_itearto,bool>类型的。所以会报错。
在这里插入图片描述
·
解决方法:

1.迭代器的拷贝函数是浅拷贝,我们不需要写,编译器自动生成的拷贝就可以用,编译器自动生成的拷贝函数只能实现普通迭代器拷贝给普通迭代器,const迭代器拷贝给const迭代器。(原理就是拷贝函数的对象类型就是调用这个函数的类型,当普通迭代器调用拷贝时,那么拷贝对象就是普通类型,当const迭代器调用拷贝时,那么拷贝对象就是const类型)
2.而我们需要的是让普通迭代器能够拷贝给const迭代器。所以我们需要自己增加拷贝函数。
3.库里的设计很妙,库里重新定义了一个iterator,作为拷贝对象,而这个iterator固定了就是普通的迭代器,不会随着调用对象而改变类型。所以当普通迭代器调用时,就会将普通iterator拷贝给它。当const迭代器调用时,就会将普通迭代器iterator拷贝给它。
4.所以我们需要对哈希表的迭代器添加拷贝构造。用普通迭代器iteartor作为拷贝对象。

struct HSIterator
{
	
	typedef  HashNode<T> Node;
	typedef HSIterator<K, T, T&, T*, KeyOfT, Hashfunc> iterator;
	typedef HSIterator<K, T,Ref,Ptr,KeyOfT, Hashfunc> Self;
	
	Node* _node;
	
	const Hash_table <K,T, KeyOfT,Hashfunc>* _pht;
	
	//没有取这个类里面的内嵌类型,就不需要加typename
	HSIterator(Node* node, Hash_table<K, T, KeyOfT,Hashfunc>* hs)//hs是普通的类型
		:_node(node)
		, _pht(hs)
	{}
	//当传过来的哈希表指针是普通指针就走上面--权限的缩小
	//重载一个,当传过来的是const修饰的哈希表指针就走这里---权限的平移
	HSIterator(Node* node, const Hash_table<K, T, KeyOfT, Hashfunc>* hs)//hs是普通的类型
		:_node(node)
		, _pht(hs)
	{}
	//普通的拷贝构造就是 const迭代器拷贝给const迭代器,普通迭代器拷贝给普通迭代器
	//而我们写的拷贝构造可以同时支持两个,当调用对象是普通迭代器时,用普通迭代器拷贝,也就是相当于赋值
	//当调用对象是const迭代器时,也是用普通迭代器拷贝。这样就支持了普通迭代器转化成const迭代器了
	HSIterator(const iterator& it)
		:_node(it._node)
		,_pht(it._pht)
	{}
	…………………………
	…………………………
	
};

这样处理后,我们再利用pair<iterator,bool>类型的构造函数,将普通迭代器转换成const迭代器。

1.先将insert返回类型利用ret接收
2.利用pair<iteartor,bool>构造将ret里的普通迭代器转换为const迭代器。

	pair<iterator,bool> insert(const K& key)
		{
			//return _ht.insert(key);//set调用的insert传回来的pair<iterator,bool>类型的,pair<iterator,bool>与pair<const_iterator,bool>是两个不同的类型
			//正常的的迭代器拷贝构造我们不需要写,因为迭代器是指针,拷贝构造就是浅拷贝,编译器自动生成的拷贝就可以
			//但是这里我们需要自己写拷贝构造; 因为需要将pair<iteartor,bool>类型,转化成pair<const_iterator,bool>类型,如何转化的呢?

			
			pair<typename Hash_table<K, K, setoft>::iterator, bool> it = _ht.insert(key);
			return pair<iterator, bool>(it.first, it.second);
		}

最后insert的改造到这里就结束了,insert改造完后,就可以实现[ ]运算符重载了。

V& operator[](const K&key)
		{
			pair<iterator, bool> ret = _ht.insert(make_pair(key,V()));
			//插入的数据是pair类型,要用make_pair构造
			return ret.first->second;
		}

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

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

相关文章

Stm32_标准库_5_呼吸灯_按键控制

Stm32按键和输出差不多 PA1为LED供给正电&#xff0c;PB5放置按键&#xff0c;按键一端接PB5,另一端接负极 void Key_Init(void){RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //APB2总线连接着GPIOBGPIO_InitStructur.GPIO_Mode GPIO_Mode_IPU;GPIO_InitStructur.…

Java下对象的序列化和反序列化(写出和读入)

代码如下&#xff1a; public class MyWork {public static void main(String[] args) throws IOException, ClassNotFoundException {//序列化File f new File("testFile/testObject.txt");ObjectOutputStream oos new ObjectOutputStream(new FileOutputStream(…

数据结构:堆的实现和堆排序及TopK问题

文章目录 1. 堆的概念和性质1.1 堆的概念1.2 堆的性质1.3 堆的作用 2. 堆的声明3. 堆的实现3.1 堆的插入3.2 删除堆顶元素3.3 利用数组建堆3.4 完整代码 4. 堆的应用4.1 堆排序4.2 TopK问题代码实现 物理结构有顺序结构存储和链式结构存储两种,二叉树理所应当也是可以顺序结构存…

实时通信协议

本文旨在简要解释如何在Web上实现客户端/服务器和客户端/客户端之间的实时通信&#xff0c;以及它们的内部工作原理和最常见的用例。 TCP vs UDP TCP和UDP都位于OSI模型的传输层&#xff0c;负责在网络上传输数据包。它们之间的主要区别在于&#xff0c;TCP在传输数据之前会打开…

26960-2011 半自动捆扎机 学习笔记

声明 本文是学习GB-T 26960-2011 半自动捆扎机. 而整理的学习笔记,分享出来希望更多人受益,如果存在侵权请及时联系我们 1 范围 本标准规定了半自动捆扎机(以下简称"捆扎机")的术语和定义、型号、型式与基本参数、技术要求、 试验方法、检验规则及标志、包装、运…

Python变量的三个特征

嗨喽&#xff0c;大家好呀~这里是爱看美女的茜茜呐 我们来看这些代码 x 10 print(x) # 获取变量的变量值 print(id(x)) # 获取变量的id&#xff0c;可以理解成变量在内存中的地址python的内置功能id()&#xff0c;内存地址不一样&#xff0c;则id()后打印的结果不一样&…

【HTML】表格行和列的合并

概述 当我们需要在 HTML 表格中展示复杂的数据时&#xff0c;行和列的合并可以帮助我们实现更灵活的布局和结构。通过合并行和列&#xff0c;我们可以创建具有更多层次和结构的表格&#xff0c;使数据更易于理解和分析。 在 HTML 表格中&#xff0c;我们可以使用 rowspan 和 …

【Spring Cloud】深入探索 Nacos 注册中心的原理,服务的注册与发现,服务分层模型,负载均衡策略,微服务的权重设置,环境隔离

文章目录 前言一、初识 Nacos 注册中心1.1 什么是 Nacos1.2 Nacos 的安装&#xff0c;配置&#xff0c;启动 二、服务的注册与发现三、Nacos 服务分层模型3.1 Nacos 的服务分级存储模型3.2 服务跨集群调用问题3.3 服务集群属性设置3.4 修改负载均衡策略为集群策略 四、根据服务…

【JUC】一文弄懂@Async的使用与原理

文章目录 1. Async异步任务概述2. 深入Async的底层2.1 Async注解2.2 EnableAsync注解2.3 默认线程池 1. Async异步任务概述 在Spring3.X的版本之后&#xff0c;内置了Async解决了多个任务同步进行导致接口响应迟缓的情况。 使用Async注解可以异步执行一个任务&#xff0c;这个…

棱镜七彩受邀参加“数字政府建设暨数字安全技术研讨会”

近日&#xff0c;为深入学习贯彻党的二十大精神&#xff0c;落实《数字中国建设整体布局规划》中关于“发展高效协同的数字政务”的要求&#xff0c;由国家信息中心主办、复旦大学义乌研究院承办、苏州棱镜七彩信息科技有限公司等单位协办的“数字政府建设暨数字安全技术研讨会…

zemax埃尔弗目镜

可以认为是一种对称设计&#xff0c;在两个双胶合透镜之间增加一个双凹单透镜 将半视场增大到30&#xff0c;所有的轴外像差维持在可以接受的水平。 入瞳直径4mm波长0.51、0.56、0.61半视场30焦距27.9mm 镜头参数&#xff1a; 成像效果&#xff1a;

Win11配置多个CUDA环境

概述 由于跑项目发现需要配置不同版本的Pytorch&#xff0c;而不同版本的Pytorch又对应不同版本的CUDA&#xff0c;于是有了在Win上装多个CUDA的打算 默认已经在电脑上装了一个CUDA 现在开始下载第二个CUDA版本&#xff0c;前面下载的操作和普通安装的几乎一样 安装CUDA CU…

CFS内网穿透靶场实战

一、简介 不久前做过的靶场。 通过复现CFS三层穿透靶场&#xff0c;让我对漏洞的利用&#xff0c;各种工具的使用以及横向穿透技术有了更深的理解。 一开始nmap探测ip端口,直接用thinkphpv5版本漏洞工具反弹shell&#xff0c;接着利用蚁剑对服务器直接进行控制&#xff0c;留下…

识别消费陷阱,反消费主义书单推荐

在消费主义无所不在的今天&#xff0c;商家是如何设置消费陷阱的&#xff1f;人们在做出消费决策时又是如何“犯错”的&#xff1f;如何才能做出更加理性的选择&#xff1f; 本书单适合对经济学、市场营销感兴趣的朋友阅读。 《小狗钱钱》 “你的自信程度决定了你是否相信自已…

kaggle_competition1_CIFAR10_Reg

一、查漏补缺、熟能生巧&#xff1a; 1.关于shutil.copy或者这个copyfile的作用和用法&#xff1a; 将对应的文件复制到对应的文件目录下 2.关于python中dict的键值对的获取方式&#xff1a; #终于明白了&#xff0c;原来python中的键_值 对的用法就是通过调用dict.keys()和…

Windows/Linux下进程信息获取

Windows/Linux下进程信息获取 前言一、windows部分二、Linux部分三、完整代码四、结果 前言 Windows/Linux下进程信息获取&#xff0c;目前可获取进程名称、进程ID、进程状态 理论分析&#xff1a; Windows版本获取进程列表的API: CreateToolhelp32Snapshot() 创建进程快照,…

GPIO的输入模式

1. GPIO支持4种输入模式&#xff08;浮空输入、上拉输入、下拉输入、模拟输入&#xff09; 1. 模拟输入 首先GPIO输出部分(N-MOS,P-MOS)是不起作用的。并且TTL施密特触发器也是不工作的。 上下拉电阻的开关都是关闭的。相当于I/o直接接在模拟输入。 模拟输入模式下&#xff…

测试开源下载模块Downloader

微信公众号“DotNet”的文章《.NET 异步、跨平台、支持分段下载的开源项目 》&#xff08;参考文献1&#xff09;介绍了GitHub中的开源下载模块Downloader的基本用法&#xff0c;本文学习Downloader的主要参数设置方式及基本用法&#xff0c;最后编写简单的测试程序进行文件下载…

[尚硅谷React笔记]——第2章 React面向组件编程

目录&#xff1a; 基本理解和使用&#xff1a; 使用React开发者工具调试函数式组件复习类的基本知识类式组件组件三大核心属性1: state 复习类中方法this指向&#xff1a; 复习bind函数&#xff1a;解决changeWeather中this指向问题&#xff1a;一般写法&#xff1a;state.htm…

毛玻璃态计算器

效果展示 页面结构组成 从上述的效果可以看出&#xff0c;计算机的页面比较规整&#xff0c;适合grid布局。 CSS3 知识点 grid 布局 实现计算机布局 <div class"container"><form class"calculator" name"calc"><input type…