数据结构——跳跃表

Posted WoLannnnn

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了数据结构——跳跃表相关的知识,希望对你有一定的参考价值。

跳跃表介绍

跳跃表(skiplist)是一种随机化的数据结构,是一种可以与平衡树媲美的层次化链表结构——查找、删除、添加等操作都可以在对数期望时间下完成,以下是一个典型的跳跃表例子:

到底有多随机,我们看了原理就知道了。

跳跃表原理

这里借助(1条消息) 跳跃表的原理和实现(Java)_CoderLucas的博客-CSDN博客_跳跃表原理和实现 中的图来讲解。

查找

假设当前我们要查找 39,我们要先从 head (第一层的第一个元素)开始查找:

  • 39 > 负无穷,继续往后查找
  • 39 > 17,继续往后查找
  • 39 < 正无穷,此时找到了第一个大于 39 的数,所以需要往后退一步,即退到最后一个小于等于 39 的数,这样方便下一层的查找. 这里是退到 17,然后跳到第二层的17,开始下一层的查找

然后从第二层开始查找:

  • 39 > 17,继续往后查找
  • 39 > 25,继续往后查找
  • 39 < 55,此时找到了第一个大于 39 的数,同样,退一步,去下一层查找,即从第三层的 25 继续查找

第三层查找:

  • 39 > 25,继续往后查找
  • 39 > 31,继续往后查找
  • 39 < 55,此时找到了第一个大于 39 的数,同样,退一步,去下一层查找,即从第四层的 31 继续查找

第四层查找:

  • 39 > 31,继续向后查找
  • 39 > 38,继续向后查找
  • 39 < 55,此时找到了第一个大于 39 的数,同样,退一步,去下一层查找,即从第四层的 38 继续查找

第五层查找:

  • 39 > 38,继续向后查找
  • 39 == 39,找到了。

插入

与查找类似,我们需要找到一个合适的插入位置,即找到最后一个小于插入值的节点(或者第一个大于插入值的节点),然后将新的节点插入。

但是,在看到跳跃表的结构时,大家可能有疑问,跳跃表这么多层是怎么创建出来的?又是何时创建出来的?每一层的节点又是怎么选取的?

在最开始,我们说了这是一种随机化的数据结构,它的随机就体现在这里,在视频 《MIT 算法导论》的第 12 集中,插入一个节点时,我们用抛硬币的方式来决定是否要将该节点往上提。由此,我们明白了何时创建多层链表,以及如何选取节点,还有一个问题是,怎么创建多层链表?

当我们将新插入的节点上提时,还需要再抛一次硬币,来判断是不是还需要将它再上提一层,如此重复,直到抛硬币的结果不再是上提。

如果上一层不存在,则创建一层,每一层都有一个负无穷和正无穷的首尾节点,以便跳跃。

删除

删除同样也和查找类似,当我们找到了要删除的节点(可以是最上层,也可以是最下层),逐层将它们从链表中删除即可,

跳跃表实现

跳跃表节点

template<class T>
struct skipEntry 
	int _key;
	T _val;
	skipEntry<T>* _prev;
	skipEntry<T>* _next;
	skipEntry<T>* _up;
	skipEntry<T>* _down;

	skipEntry(int key = 0, T val = T())
		:_key(key)
		,_val(val)
		, _prev(nullptr)
		, _next(nullptr)
		, _up(nullptr)
		, _down(nullptr)
	
;

跳跃表

跳跃表数据结构主要含有的属性有:

	int _n; // 总元素数量

	// 跳跃表首尾节点
	skipEntry<T>* _head;
	skipEntry<T>* _tail;

	unsigned _seed; // 随机种子,生成随机数,判断是否需要将插入节点上升

跳跃表初始化

这里再次借助上面文章的图,来展现一个初始化跳跃表的模型(博主的图画的是真好):

基本操作

插入

因为 head 始终是指向第一层的负无穷节点,所以我们都是从 head 开始查找。如果 key 已经存在,则更新 val,否则执行插入

假设我们现在要在下图中已有的跳跃表中插入 23,首先找到最下层的最后一个小于 23 的节点,然后插入,并判断是否要将其上提。

如果需要上提,则从插入位置开始,向前查找第一个含有上层节点的元素p,然后创建一个新节点,将其插入到上一层。如果没有上一层,则创建一层。

查找与删除

查找与删除和上面提到的都差不多,这里就不赘述了。

随机化的方法

博主这里使用的是 C++ 的 rand 与 srand 产生随机数,然后判断奇偶来决定是否需要上提,rand 与 srand 的使用读者可以自行查阅,使用很简单。

实现代码

博主实现的很简单,只完成了基本的增删改查

#include <cstdlib>
#include <ctime>
#include <limits.h>

// 跳跃表节点
template<class T>
struct skipEntry 
	int _key;
	T _val;
	skipEntry<T>* _prev;
	skipEntry<T>* _next;
	skipEntry<T>* _up;
	skipEntry<T>* _down;

	skipEntry(int key = 0, T val = T())
		:_key(key)
		,_val(val)
		, _prev(nullptr)
		, _next(nullptr)
		, _up(nullptr)
		, _down(nullptr)
	
;

template<class T>
class skiplist 
private:
	int _n; // 总元素数量

	// 跳跃表首尾节点
	skipEntry<T>* _head;
	skipEntry<T>* _tail;

	unsigned _seed; // 随机种子,生成随机数,判断是否需要将插入节点上升

	// 生成随机数
	int generateRandomNum()
	
		_seed = time(0);
		srand(_seed);
		return rand() % 2;
	

public:
	skiplist()
		:_n(0)
		,_head(new skipEntry<T>(INT_MIN))
		,_tail(new skipEntry<T>(INT_MAX))
		,_seed(time(0))
	
		_head->_next = _tail;
		_tail->_prev = _head;
	

private:
	// 查找,如果找到了返回 key 值,如果没找到返回小于它的前一个节点
	skipEntry<T>* _findEntry(int key)
	
		skipEntry<T>* cur = _head;

		// 从 _head 开始查找
		while (1)
		
			// 找到第一个大于 key 的节点
			while (key >= cur->_key)
			
				cur = cur->_next;
			

			cur = cur->_prev; // 返回到第一个小于等于 key 的节点
			// 判断有没有下一层
			if (cur->_down)
			
				cur = cur->_down;
				continue;
			

			// 返回最后一个小于key的节点
			return cur;

			break;
		

		return nullptr;
	

public:
	bool insert(int key, T val)
	
		skipEntry<T>* insertpos = _findEntry(key);
		if (insertpos->_key == key) // 已存在
		
			insertpos->_val = val;
			return true;
		

		skipEntry<T>* newelem = new skipEntry<T>;
		if (newelem == nullptr)
			return false;

		skipEntry<T>* next = insertpos->_next;
		newelem->_key = key;
		newelem->_val = val;
		newelem->_prev = insertpos;
		newelem->_next = next;

		insertpos->_next = newelem;
		next->_prev = newelem;

		++_n;

		// 判断当前节点是否需要上升
		while (generateRandomNum() == 1) // 需要上升
		
			// 1.找到 newelem 前面第一个含有上升节点的元素 p,如果没有,则创建新的一层
			// 2.创建一个与 newelem 相同的节点,连接 newelem 与 该节点,将该节点插到上层
			// 3.继续判断是否要上升

			// 1.找到 newelem 前面第一个含有上升节点的元素 p,如果没有,则创建新的一层
			skipEntry<T>* p = newelem->_prev;
			while (p && p->_up == nullptr)
			
				p = p->_prev;
			

			if (p == nullptr) // 创建一层
			
				p = _head;

				// 需要 INT_MIN, INT_MAX 两个节点作为该层的首尾节点
				skipEntry<T>* curLevelHead = new skipEntry<T>(INT_MIN);
				skipEntry<T>* curLevelTail = new skipEntry<T>(INT_MAX);
				curLevelHead->_next = curLevelTail;
				curLevelTail->_prev = curLevelHead;

				_head->_up = curLevelHead;
				curLevelHead->_down = _head;
				_tail->_up = curLevelTail;
				curLevelTail->_down = _tail;

				_head = curLevelHead;
				_tail = curLevelTail;

			

			// 2.创建一个与 newelem 相同的节点,连接 newelem 与 该节点,将该节点插到上层
			skipEntry<T>* sameelem = new skipEntry<T>(key, val);
			sameelem->_down = newelem;
			newelem->_up = sameelem;

			p = p->_up; 
			skipEntry<T>* pnext = p->_next;
			sameelem->_prev = p;
			p->_next = sameelem;
			sameelem->_next = pnext;
			pnext->_prev = sameelem;

			// 3.继续判断是否要上升
			newelem = sameelem;
		

		return true;
	

	bool erase(int key)
	
		// 从下至上删除

		skipEntry<T>* cur = _findEntry(key);
		if (cur->_key != key)
			return false;

		while (cur)
		
			cur->_prev->_next = cur->_next;
			cur->_next->_prev = cur->_prev;

			skipEntry<T>* up = cur->_up;
			delete cur;
			cur = up;
		

		return false;
	

	void* find(int key)
	
		skipEntry<T>* ret = _findEntry(key);
		return ret->_key == key ? (void*)ret->_val : nullptr;
	

	void clear()
	
		skipEntry<T>* first = _head;
		skipEntry<T>* last = _tail;

		while (first->_down)
		
			first = first->_down;
		

		while (last->_down)
		
			last = last->_down;
		

		skipEntry<T>*cur = first->_next;
		while (cur != last)
		
			erase(cur->_key);
			cur = cur->_next;
		

		// 删除每一层的首尾元素
		while(first)
		
			skipEntry<T>* up = first->_up;
			delete first;
			first = up;
		

		while (last)
		
			skipEntry<T>* up = last->_up;
			delete last;
			last = up;
		
	

	~skiplist()
	
		clear();
	
;

以上是关于数据结构——跳跃表的主要内容,如果未能解决你的问题,请参考以下文章

Redis跳跃表

数据结构——跳跃表

Redis中的跳跃表

Redis数据结构之跳跃表

redis源码跳跃表(zskiplist)

redis 系列7 数据结构之跳跃表