哈希表原理,以及unordered_set/和unordered_map的封装和迭代器的实现

news2025/1/23 12:04:51

哈希表

  • unordered系列
  • unordered_set和unordered_map的使用
  • 哈希
    • 哈希概念
    • 哈希冲突
    • 哈希函数
    • 闭散列
    • 开散列
    • 哈希表的扩容
    • 哈希表源码(开散列和闭散列)
  • 封装unordered_set/和unordered_map,以及实现迭代器
    • 节点定义
    • unordered_set定义
    • unordered_map定义
    • 哈希表实现
    • 迭代器实现

unordered系列

C++98中引入了map和set,这两种类型底层数据结构都是红黑树,不管是插入、删除、查询等操作都是log2n ,这个时间复杂度已经非常快了,但是有没有一种数据结构能通过值直接定位到存储位置,所以C++11中引入了unordered系列,这次重点讨论unordered_set/unordered_map,unordered底层就是用的哈希表,也可以称为散列表。接下来我会详细剖析哈希表的原理,以及哈希表如何被封装成unordered_set和unordered_map。

unordered_set和unordered_map的使用

其实unordered_set和unordered_map使用上基本没有区别。

void test_unordered_set1()
{
	unordered_set<int> s;
	s.insert(1);
	s.insert(3);
	s.insert(2);
	s.insert(7);
	s.insert(2);

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

	for (auto e : s)
	{
		cout << e << " ";
	}
	cout << endl;
}

在这里插入图片描述

void test_unordered_map()
{
	string arr[] = { "苹果", "菠萝", "葡萄", "菠萝", "哈密瓜", "香蕉", "苹果", "香蕉", "梨", "西红柿", "香蕉", "菠萝" };
	map<string, int> countMap;
	for (auto& e : arr)
	{
		countMap[e]++;
	}

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

在这里插入图片描述
在这里插入图片描述
这就是unordered最常用的函数,和map/set没什么区别,如果想详细了解这些接口的使用,可以通过这个C++的网址进行查询:链接: cplusplus。

哈希

哈希概念

了解哈希表之前我先浅浅的提问一个问题,给你一个字符串,统计每个字母出现的次数(默认都是小写),应该怎么统计?

这是一道非常简单的题目,可能方法很多,但是我们可以采用比较快捷的方式,采用映射的方式,总共有26的英文字母,开一个int类型大小为26的数组,存储a~z出现的次数,遍历字符串,每个字母减去a就是数组对应的下标,只需要通过数组下标内的值就可以完成对字母出现次数的统计。
在这里插入图片描述
这问题的本质其实就是通过一种特定的映射方式,一次就可以找到相对应的位置,哈希表(也称散列表)也是通过这种原理,通过特定的函数,计算key的唯一存储位置。

哈希概念:以不经过任何比较,一次直接从表中得到要搜索的元素。如果构造一种存储结构,通过某种函数(hashFunc)使元素的存储位置与它的关键码之间能够建立一一映射的关系,那么在查找时通过该函数可以很快找到该元素。

该方式即为哈希(散列)方法,哈希方法中使用的转换函数称为哈希(散列)函数,构造出来的结构称为哈希表(Hash Table)(或者称散列表)

例如:数据集合{1,7,6,4,5,9};
哈希函数设置为:hash(key) = key % capacity( 除留余数法,哈希函数介绍); capacity为存储元素底层空间总的大小。
在这里插入图片描述
用该方法进行搜索不必进行多次关键码的比较,所以哈希表的查询速度是O(1)的,非常快。

哈希冲突

上面通过key除以capacity求得了存储位置,这种方式有什么问题呢?答案是肯定有,例如在插入一个44,通过44%10=4,可是4的位置已经有了元素,这种情况称为哈希冲突。

哈希冲突可以用开散列和闭散列的方式去解决。

哈希函数

引起哈希冲突的一个原因可能是:哈希函数设计不够合理。
哈希函数设计原则:

  • 哈希函数的定义域必须包括需要存储的全部关键码,而如果散列表允许有m个地址时,其值域必须在0到m-1之间
  • 哈希函数计算出来的地址能均匀分布在整个空间中
  • 哈希函数应该比较简单

常见的哈希函数:

  1. 直接定址法–(常用)
    取关键字的某个线性函数为散列地址:Hash(Key)= A*Key + B
    优点:简单、均匀
    缺点:需要事先知道关键字的分布情况
    使用场景:适合查找比较小且连续的情况
  2. 除留余数法–(常用)
    设散列表中允许的地址数为m,取一个不大于m,但最接近或者等于m的质数p作为除数,
    按照哈希函数:Hash(key) = key% p(p<=m),将关键码转换成哈希地址。

我们最常用的就是除留余数法,所以其他哈希函数了解就好了。

还有一个问题就是上面举例中key都是整数,所以可以拿key直接%数组大小,得到存储位置,但是如果是字符串或者浮点数或者自定义类型呢?这个就需要哈希函数把相应的类型转换成int或者unsigned int。

例如:
字符串需要怎么转换成整形呢?或许很多人想到把他们相加,可是如果只是相加的话,大量字符串只是排列组合不一样也会映射到同一个存储位置,闭散列会发生大量踩踏,开散列就是把数据都挂到一个桶内,所以采用下列这种方式。加每个字符之后乘31,也可以乘以131、1313、13131、131313… 。本算法由于在Brian Kernighan与Dennis Ritchie的《The C Programming Language》一书被展示而得 名,是一种简单快捷的hash算法,也是Java目前采用的字符串的Hash算法(累乘因子为31)。

//特化
template<>
struct HashFunc<string>
{
	// BKDR
	size_t operator()(const string& s)
	{
		size_t hash = 0;
		for (auto ch : s)
		{
			hash += ch;
			hash *= 31;
		}

		return hash;
	}
};

闭散列

也叫开放定址法,当发生哈希冲突时,如果哈希表未被装满,说明在哈希表中必然还有空位置,那么可以把key存放到冲突位置中的“下一个” 空位置中去。那如何寻找下一个空位置呢?

例如数组中现在需要插入元素44,先通过哈希函数计算哈希地址,hashAddr为4,因此44理论上应该插在该位置,但是该位置已经放了值为4的元素,即发生哈希冲突。
在这里插入图片描述
解决方案

  • 线性探测:从发生冲突的位置开始,依次向后探测,直到寻找到下一个空位置为止。

在这里插入图片描述

  • 二次探测:依次检测第i2个位置是否为空

线性探测的缺陷是产生冲突的数据堆积在一块,这与其找下一个空位置有关系,因为找空位置的方式就是挨着往后逐个去找,因此二次探测为了避免该问题,找下一个空位置的方法为: H i H_i Hi = ( H 0 H_0 H0 + i 2 i^2 i2 )% m, 或者: H i H_i Hi = ( H 0 H_0 H0 - i 2 i^2 i2 )% m。其中:i = 1,2,3…, H 0 H_0 H0是通过散列函数Hash(x)对元素的关键码 key 进行计算得到的位置,m是表的大小。

闭散列的删除:
采用闭散列处理哈希冲突时,不能随便物理删除哈希表中已有的元素,若直接删除元素会影响其他元素的搜索。比如删除元素4,如果直接删除掉,44查找起来可能会受影响。因此线性探测采用标记的伪删除法来删除一个元素。

enum Status {
	EMPTY,//位置为空
	EXIST,//位置不为空
	DELETE//删除状态
};

总结

线性探测和二次探测是解决哈希冲突的两种方法。

线性探测,顾名思义,是一种按线性方式探测的方法。如果发生了哈希冲突,它会尝试顺序地检查下一个哈希槽,直到找到一个空位或者检查完了整个哈希表。但是线性探测容易导致聚集性冲突,即相邻位置都被占用的情况。如果哈希表的利用率比较高,那么冲突解决的效率也会越来越低。

而二次探测是一种按照平方探测的方式解决哈希冲突的方法。相对于线性探测,它探测的跨度更大,可以充分利用哈希表的空间。当一个位置已经被占用时,它会依次探测第 1 2 1^2 12、第 2 2 2^2 22、第 3 2 3^2 32……个位置,直到找到一个空位或者找遍整个哈希表。但是,二次探测容易导致“二次探测陷阱”,使得某些哈希槽永远无法被探测到。

综上,线性探测和二次探测各有优劣,并且适用于不同的场景。在哈希表的容量比较大,利用率比较低的情况下,二次探测效率更高。但如果哈希表的利用率比较高,那么线性探测可能更加适用。在实际应用中,可以通过选择不同的哈希函数和调节哈希表的容量来优化哈希表的性能。

闭散列学习了解即可,实际运用中开散列是用的比较多的。

闭散列源码实现:

	enum Status {
		EMPTY,
		EXIST,
		DELETE
	};

	template<class K, class V>
	struct HashData {
		pair<K, V> _kv;
		Status status = EMPTY;
	};

	template<class K, class V>
	class HashTable {
		typedef HashData<K, V> Node;
	public:
		Node* Find(const K& key)
		{
			if (_table.size() == 0)
			{
				return nullptr;
			}
			int hashi = key % _table.size();
			int index = hashi;
			while (_table[index]._kv.first != key)
			{
				index++;
				index %= _table.size();
				if (index == hashi)
				{
					return nullptr;
				}
			}
			if (_table[index].status == EXIST)
			{
				return &_table[index];
			}
			else {
				return nullptr;
			}

		}
		bool Erase(const K& key)
		{

			Node* ret = Find(key);
			if (ret)
			{
				ret->status = DELETE;
				return true;
			}
			else
			{
				return false;
			}
		}
		bool insert(const pair<K, V>& kv)
		{
			if (Find(kv.first))
			{
				return false;
			}
			if (_table.size() == 0 || n * 10 / _table.size() == 7) {
				//扩容
				size_t newsize = _table.size() == 0 ? 10 : _table.size() * 2;
				HashTable<K, V> newtable;//扩容需要重新开辟空间
				newtable._table.resize(newsize);
				for (auto& data : _table) {
					if (data.status == EXIST)
					{
						newtable.insert(data._kv);
					}
				}
				_table.swap(newtable._table);
			}
			int hashi = kv.first % _table.size();
			int index = hashi;
			while (_table[index].status == EXIST) {
				index++;
				index %= _table.size();
				if (index == hashi)
				{
					break;
				}
			}
			_table[index]._kv = kv;
			_table[index].status = EXIST;
			n++;
			return true;
		}
	private:
		vector<Node> _table;
		int n = 0;
	};

开散列

闭散列有许多的弊端,让哈希表实际当中并不是那么理想,开散列也是解决哈希冲突的一种方式,也是实际当中用的比较多的一种方法。
开散列也成为哈希桶,数组里面存储的是单链表,发生了哈希冲突之后,不会向后找空位置,而是插入到存储位置的链表尾部。

template<class K, class V>
	struct HashNode
	{
		HashNode<K, V>* _next;
		pair<K, V> _kv;

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

在这里插入图片描述
所以我们通过哈希函数就可以找到存储位置,插入单链表节点就好,想要删除也是单链表的删除。

开散列(哈希桶)实现源码:

	template<class K,class V>
	struct HashData
	{
		pair<K, V> _kv;
		HashData<K, V>* next;
		HashData(const pair<K,V>& kv)
			:_kv(kv)
			,next(nullptr)
		{}
	};
	template<class K,class V>
	class HashTable
	{
		typedef HashData<K, V> Node;
	public:
		~HashTable()
		{
			if (_table.size() > 0)
			{
				for (auto& cur : _table)
				{
					while (cur)
					{
						Node* next = cur->next;
						delete cur;
						cur = next;
					}
				}
			}
		}
		bool Erase(const K& key)
		{
			int hashi = key % _table.size();
			Node* cur = _table[hashi];
			Node* prev = nullptr;
			while (cur)
			{
				if (cur->_kv.first == key)
				{
					if (prev == nullptr)
					{
						_table[hashi] = cur->next;
						delete cur;
					}
					else
					{
						prev->next = cur->next;
						delete cur;
					}
					cur = nullptr;
					return true;
				}
				prev = cur;
				cur = cur->next;
			}
			return false;

		}
		Node* Find(const K& key)
		{
			if (_table.size() == 0)
			{
				return nullptr;
			}
			int hashi = key % _table.size();
			Node* cur = _table[hashi];
			while (cur)
			{
				if (cur->_kv.first == key)
				{
					return cur;
				}
				cur = cur->next;
			}
			return nullptr;
		}
		bool Insert(const pair<K, V>& kv)
		{
			if (Find(kv.first))
			{
				return false;
			}
			if (_table.size() == 0 || n / _table.size() == 1)
			{
				//扩容
				size_t newsize = _table.size() == 0 ? 10 : _table.size() * 2;
				vector<Node*> newtable;
				newtable.resize(newsize,nullptr);

				for (auto& cur : _table)
				{
					while (cur)
					{
						Node* next = cur->next;
						size_t i = cur->_kv.first % newtable.size();
						cur->next = newtable[i];
						newtable[i] = cur;
						cur = next;
					}
				}
				_table.swap(newtable);
			}
			Node* newnode = new Node(kv);
			int hashi = kv.first % _table.size();
			newnode->next = _table[hashi];
			_table[hashi] = newnode;
			n++;
			return true;
		}
	private:
		vector<Node*> _table;
		int n = 0;
	};

哈希表的扩容

引入载荷因子
散列表的载荷因子定义为: α =填入表中的元素个数/散列表的长度

α是散列表装满程度的标志因子。由于表长是定值,α与“填入表中的元素个数”成正比,所以,a越大,表明填入表中的元素越多,产生冲突的可能性就越大;反之,α越小,标明填入表中的元素越少,产生冲突的可能性就越小。实际上,散列表的平均查找长度是载荷因子α的函数,只是不同处理冲突的方法有不同的函数。
对于开放定址法,荷载因子是特别重要因素,应严格限制在0.7-0.8以下。超过0.8,查表时的CPU缓存不命中(cachemissing)按照指数曲线上升。因此,一些采用开放定址法的hash库,如Java的系统库限制了荷载因子为0.75,超过此值将resize散列表。
哈希桶的载荷因子控制在1即可。

哈希表源码(开散列和闭散列)

源码链接

封装unordered_set/和unordered_map,以及实现迭代器

哈希表的原理说明已经讲解清楚了,接下来需要对源码稍微改动一下,对其封装。所有代码拿来就能用。代码的含义会在注释里面讲解清楚。

节点定义

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

unordered_set定义

	template<class K,class Hash = HashFun<K>>//HashFun是保证传进来的类型能转换成整型,如果key是自定义类型,HashFun需要自己定义,如果是string就用哈希函数中介绍的方法
	class unorder_set
	{
		struct SetOfT {
			size_t operator()(const K& key)
			{
				return key;
			}
		};
	public:
		//由于key不可修改所以const迭代器就是普通迭代器
		typedef typename HashTable<K, K, SetOfT, Hash>::const_iterator iterator;
		
		//全部调用哈希表中的接口
		iterator begin() const
		{
			return ht.begin();
		}
		iterator end() const
		{
			return ht.end();
		}
		bool insert(const K& key)
		{
			return ht.Insert(key);
		}
		bool Erase(const K& key)
		{
			return ht.Erase(key);
		}
	private:
		HashTable<K,K, SetOfT,Hash> ht;
	};

unordered_map定义

	#pragma once
#include "hash_table.h"
namespace JRG {

	template<class K,class V,class Hash = HashFun<K>>
	class unorder_map
	{
		struct MapOfT {
			size_t operator()(const pair<K,V>& key)
			{
				return key.first;
			}
		};
	public:
		typedef typename HashTable<K, pair<const K, V>, MapOfT, Hash>::iterator iterator;
		typedef typename HashTable<K, pair<const K, V>, MapOfT, Hash>::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>& data)
		{
			return ht.Insert(data);
		}
		bool erase(const K& data)
		{
			return ht.Erase(data);
		}
	private:
		HashTable<K,pair<const K,V> , MapOfT, Hash> ht;
	};
}

哈希表实现

这是哈希表改动之后的源码,我就直接在代码上面讲解,默认用哈希桶的方式实现,详情请看注释。

#pragma once
//哈希表实现
template<class K,class T,class KeyOfT,class Hash>
class HashTable
{
private:
	vector<Node*> _table;
	int n = 0;//存储元素的个数
public:
	typedef HashNode<T> Node;
	//迭代器需要访问哈希表,把迭代器定义为友元类。
	template<class K, class T, class Ref, class Ptr, class KeyOfT, class Hash>
	friend struct _HashIterator;


	typedef _HashIterator<K, T, T&, T*, KeyOfT, Hash> iterator;
	typedef _HashIterator<K, T, const T&, const T*, KeyOfT, Hash> const_iterator;

	//
	Hash hash;
	KeyOfT keyOfT;
	
	iterator begin()
	{
		Node* cur=nullptr;
		for (size_t i = 0; i < _table.size(); i++)
		{
			cur = _table[i];
			if (cur)
			{
				break;
			}
		}
		return iterator(cur, this);
	}
	iterator end()  
	{
		return iterator(nullptr, this);
	}
	~HashTable()
	{
		if (_table.size() > 0)
		{
			for (auto& cur : _table)
			{
				while (cur)
				{
					Node* next = cur->next;
					delete cur;
					cur = next;
				}
			}
		}
	}
	//每次扩容数组大小都为质数,这是哈希表每次扩容之后的大小,每次从数组获取下次扩容的大小。
	size_t GetNextPrime(size_t prime)
	{
		// SGI
		static const int __stl_num_primes = 28;
		static const unsigned long __stl_prime_list[__stl_num_primes] =
		{
			53, 97, 193, 389, 769,
			1543, 3079, 6151, 12289, 24593,
			49157, 98317, 196613, 393241, 786433,
			1572869, 3145739, 6291469, 12582917, 25165843,
			50331653, 100663319, 201326611, 402653189, 805306457,
			1610612741, 3221225473, 4294967291
		};

		size_t i = 0;
		for (; i < __stl_num_primes; ++i)
		{
			if (__stl_prime_list[i] > prime)
				return __stl_prime_list[i];
		}

		return __stl_prime_list[i];
	}
	//根据哈希桶原理,传入key,可以对其进行删除
	bool Erase(const K& key)
	{

		size_t hashi = hash(key) % _table.size();
		Node* prev = nullptr;
		Node* cur = _table[hashi];
		while (cur)
		{
			if (keyOfT(cur->_data) == key)
			{
				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;
	}
	//查找key
	iterator Find(const K& key)
	{
		if (_table.size() == 0)
		{
			return end();
		}
			
		size_t hashi = hash(key) % _table.size();
		Node* cur = _table[hashi];
		while (cur)
		{
			if (keyOfT(cur->_data) == key)
			{
				return iterator(cur, this);
			}

			cur = cur->_next;
		}

		return end();
	}
	//根据哈希桶的原理实现插入函数,扩容逻辑也是在insert内部完成
	pair<iterator, bool> Insert(const T& data)
	{
		//如果要插入的值已经存在,返回迭代器
		iterator it = Find(keyOfT(data));
		if (it != end())
		{
			return make_pair(it, false);
		}

		//需要扩容
		if (_table.size() == 0 || n / _table.size() == 1)
		{
			vector<Node*> newtable;
			newtable.resize(GetNextPrime(_table.size()), nullptr);

			for (auto& cur : _table)
			{
				while (cur)
				{
					Node* next = cur->_next;
					size_t i = hash(keyOfT(cur->_data)) % newtable.size();
					cur->_next = newtable[i];
					newtable[i] = cur;
					cur = next;
				}
			}
			_table.swap(newtable);
		}
		//找到对应的哈希桶,插入链表节点
		Node* newnode = new Node(data);
		int hashi = keyOfT(data) % _table.size();
		newnode->_next = _table[hashi];
		_table[hashi] = newnode;
		n++;
		return make_pair(it, true);
	}

};

迭代器实现

//提前声明,因为迭代器中要传入哈希表
template<class K, class T, class KeyOfT, class Hash>
class HashTable;

//迭代器的实现,哈希表的迭代器是单向迭代器,所以不支持--。
template<class K, class T, class Ref,class Ptr,class KeyOfT, class Hash>
struct _HashIterator
{
	typedef  HashNode<T> Node;
	typedef  HashTable<K, T, KeyOfT,Hash> Table;
	typedef _HashIterator<K, T, Ref, Ptr, KeyOfT, Hash> Self;
	typedef _HashIterator<K, T, T&, T*, KeyOfT, Hash> iterator;
	Node* _node;
	const Table* _Table; //迭代器被哈希表定为友元类,所以可以直接访问哈希表的成员变量
	_HashIterator(Node* node, const Table* table)
		:_node(node)
		, _Table(table)
	{}
	//由于unordered_set只有const迭代器,所以需要用普通迭代器构建const迭代器
	_HashIterator(const iterator& it)
		:_node(it._node)
		, _Table(it._Table)
	{}
	Ref operator*()
	{
		return _node->_data;
	}
	Ptr operator->()
	{
		return &_node->_data;
	}
	bool operator!=(const Self& s)
	{
		return _node != s._node;
	}
	Self& operator++()
	{
		if (_node->next)
		{
			_node = _node->next;
			return *this;
		}
		else {	//证明这个桶内节点访问完了,寻找下一个不为空的桶
			Hash hash;//这是Hash类重载的函数体,负责保证可以把不是整数的类型转换成整数。
			KeyOfT keyoft;//这是KeyOfT类重载(),如果是set就返回key,map返回kv.frist。
			size_t hashi = hash(keyoft(_node->_data)) % _Table->_table.size(); //除留余数法计算当前桶的位置
			hashi++;
			while (hashi < _Table->_table.size())//保证不越界的情况下查找下一个不为空的桶
			{

				if (_Table->_table[hashi])//找到不为空的桶,返回迭代器本身
				{
					_node = _Table->_table[hashi];
					return *this;
				}
				hashi++;
			}
			//哈希表遍历结束,没有下一个节点。
			_node = nullptr;
			return *this;
		}
		
	}
};

所有源码链接:link

这就是哈希表的全部内容,有什么问题可以直接私信我,码字不易,觉得还不错点个赞和关注吧。

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

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

相关文章

Python3 flask-socketio 整合vue

1. 前端说明 前端追加了vue-socket.io的依赖 更新package.json及package-lock.json后&#xff0c;需要补充库 2. 后端说明 服务端需要安装的python包 suse python 3.6.12 Flask_SocketIO-4.3.1-py2.py3-none-any.whl python_socketio-4.6.0-py2.py3-none-any.whl python_…

在PPT里如何做出晶莹剔透的高级水泡感出来呢

开篇 很多PPT特别是一些TED演讲类PPT,它的首页开篇给人以强烈的冲击感。比如说上面这张PPT,笔者就卖出过一页800元的效果。它的高级高级在以下两个地方: 水晶感;透明感这是时下非常流行的一种TED场景式演讲专用的PPT,适用于一些大厂的CTO、CEO都大量使用这种效果。 今天…

shell脚本编写辅助命令

目录 一、echo 命令 二、字符串相关操作 1.截取字符串 2.获取字符串长度 3.字符串追加字符 4.从开头或结尾删除字符串指定格式内容 三、随机数 1.使用 $RANDOM 2.指定RANDOM变量的范围 &#xff08;1&#xff09;从0开始的范围 &#xff08;2&#xff09;从指定数始…

35、git的使用

一、git简介 git是一款免费、开源的版本控制系统&#xff0c;用于高效地处理任何或大或小的项目。 作用&#xff1a; 文件存档备份 文件版本管理 多人协同合作&#xff08;自动合并&#xff09; 二、git的三个区域 工作区&#xff1a;处理工作的区域 暂存区&#xff1a;临时存放…

Shell脚本实现数组冒泡排序等简单算法排序

目录 一、冒泡排序 1.简介 2.基本思想 3.算法思路 4.shell脚本实现 二、选择排序 1.简介 2.基本思想 3.shell脚本实现 三、插入排序 1.算法思路 2.shell脚本实现 四、反转排序 1.作用 2.shell脚本实现 一、冒泡排序 1.简介 类似气泡上涌的动作&#xff0c;会将…

Java入门之学习随记(三)

一. 栈内存和堆内存 堆内存:存放"引用数据类型的数据"和"new出来的对象",注意-创建出来的对象只包含各自的成员变量,不包括成员方法. 栈内存:存放"基本数据类型的数据","引用数据类型的变量名"以及"对象的引用",但是引用数…

k8s之ReplicaSet回收pod的优先级

ReplicaSet syncReplicaSet的逻辑&#xff0c;首先找到rs下相关的pod // 从informer中获取当前ns下所有的podallPods, err : rsc.podLister.Pods(rs.Namespace).List(labels.Everything())if err ! nil {return err}// 忽略不健康的podfilteredPods : controller.FilterActiveP…

CentOS桥接模式下设置静态IP并解决java.net.ConnectException: Connection timed out: connect

一、前言 最近在配置服务器&#xff0c;DHCP模式下IP地址不固定&#xff0c;每次SSH远程登录连接不上&#xff0c;都要查看新动态分配的IP地址重新配置&#xff0c;感觉些许麻烦&#xff0c;于是给机器配置了静态固定IP。 动态主机配置协议DHCP&#xff08;Dynamic Host Confi…

Shell脚本练习题(附详细解题过程)

目录 一、利用for循环打印99乘法表 二、十进制转二进制 三、将十进制ip地址转换为二进制格式 四、检测某个网段中存活的主机并输出其ip地址 五、检查文件中用户名是否存在并提示创建用户和设置密码 六、检查httpd服务并开启 七、根据百米赛跑成绩判断结果 八、随机数…

栈和队列(数据结构刷题)[一]-python

文章目录 前言一、原理介绍二、用栈实现队列1.操作2.思路 三、关于面试考察栈里面的元素在内存中是连续分布的么&#xff1f; 前言 提到栈和队列&#xff0c;大家可能对它们的了解只停留在表面&#xff0c;再深入一点&#xff0c;好像知道又好像不知道的感觉。本文我将从底层实…

Django----------模板、静态文件、案例(城市天气预报)、请求和响应

目录 1.templates模板 2.静态文件 1.static目录 2.引用静态文件 1.方式一&#xff1a;直接引用 2.方式二&#xff1a;头部及内部引用 3. 模板语法 1.取内容 2.取下标 3.for循环 4.利用字典 5.列表里套字典 6.if条件语句 7.总结 4.案例&#xff08;城市天气预…

CH573-01-GPIO-LED——RISC-V内核BLE MCU快速开发教程

1. 新建工程 1) NEW Project 点击“File->New->MounRiver Project”&#xff1a; 2) finish 选择CH573F的裸机开发工程模板&#xff0c;如下图&#xff0c;然后点击“finish” 3) 编译检查 4) 精简代码 打开工程目录下的./src/main.c文件&#xff0c;修改删掉生成的串口测…

动态规划III (买股票-121、122、123、188)

CP121 买股票的最佳时机 题目描述&#xff1a; 给定一个数组 prices &#xff0c;它的第 i 个元素 prices[i] 表示一支给定股票第 i 天的价格。你只能选择 某一天 买入这只股票&#xff0c;并选择在 未来的某一个不同的日子 卖出该股票。设计一个算法来计算你所能获取的最大利…

Vue 中的几种动画效果

Vue 中的动画效果 在 Vue 中&#xff0c;动画效果是非常常见的交互方式。它可以为用户提供更加生动的交互体验&#xff0c;增强用户的参与感和满意度。在本文中&#xff0c;我们将探讨 Vue 中的动画效果的基本原理和用法&#xff0c;并给出一些实例代码来帮助读者更好地理解。…

『手撕 Mybatis 源码』06 - Mapper 代理方式初始化

Mapper 代理方式初始化 首先修改一下 SqlSession 获取代理对象方式&#xff0c;即通过 getMapper() 来拿到动态代理对象 public class MybatisTest {/*** 问题1&#xff1a;<package name"com.itheima.mapper"/> 是如何进行解析的&#xff1f;* 解答&#xf…

算法刷题-数组-有序数组的平方

977.有序数组的平方 力扣题目链接 给你一个按 非递减顺序 排序的整数数组 nums&#xff0c;返回 每个数字的平方 组成的新数组&#xff0c;要求也按 非递减顺序 排序。 示例 1&#xff1a; 输入&#xff1a;nums [-4,-1,0,3,10] 输出&#xff1a;[0,1,9,16,100] 解释&#…

clang到底是什么?gcc和clang到底有什么区别?

最近发现自己对 GNU GCC 和 Clang 的区别不太清楚&#xff0c;影响到一些实现和学习&#xff0c;所以趁这两天有空好好研究了一下。 在这个研究过程中&#xff0c;我发现很多问题其实源自于语言&#xff08;不是指编程语言&#xff0c;而是中文和英文翻译的失真&#xff09;和…

前端前端学习不断

卷吧卷吧...&#xff0c;这东西什么时候是个头啊……

智能指针(2)

智能指针&#xff08;2&#xff09; shared_ptr(共享型智能指针)基础知识特点引用计数器共享型智能指针结构理解 shared_ptr仿写删除器类计数器类shared_ptr类使用以及仿写代码的理解 循环引用_Weaks 初始化智能指针的方法 shared_ptr(共享型智能指针) 基础知识 在java中有一…

chatgpt赋能python:Python如何判断输入的字符——基础教程与实例

Python如何判断输入的字符——基础教程与实例 时至今日&#xff0c;互联网已经成为人们获取信息的重要途径&#xff0c;而搜索引擎优化&#xff08;SEO&#xff09;则是网站重要的推广手段之一。而Python作为一种高级编程语言&#xff0c;在实现SEO时也有很大的优势&#xff0…