一个神奇的数据结构

本文介绍了如何结合哈希表和链表实现一个高效的数据结构,以解决链表查找效率低的问题。在缓存和其他场景中,哈希表常用于提供常数级查找。然而,双重数据结构可能带来拷贝和同步问题。通过定义包含val、next和prev指针的双向链表节点,并使用STL的哈希表作为索引,实现了高效查找和删除。关键在于哈希表根据key进行哈希操作,直接定位链表节点,避免了多次对象拷贝。

Linklist HashTable

https://github.com/ez8-co/linked_hash/blob/master/linked_hash.hpp

今天在群里有一位群友分享了一个数据结构。通常大家都会使用到数据结构链表。但是链表由于不具备常数级的查找复杂度,有时候需要借助hashTable来做索引方便查找。这常用与缓存中。(其实,队列,二叉树这些都可以借助哈希表来提高其常数级的查找。)

但是使用双数据结构会带来拷贝开销更大的问题,以及同步问题。但是哈希表具有常数级别查找和删除的效率。最适合作为辅助索引使用。

这里贴出部分代码分析。

1. 定义链表节点,val存放数值,另外具有next和prev两个指针,非常标准的双向链表节点。

template<class value_type>
struct lh_entry
{
	lh_entry(lh_entry* p = NULL, lh_entry* n = NULL) : val(), prev(p), next(n) {}
	lh_entry(const value_type& v, lh_entry* p, lh_entry* n) : val(v), prev(p), next(n) {}

	value_type val;
	lh_entry *prev, *next;
};

2. 定义链表哈希集合,仅贴出数据结构。

template<class _Kty, class HashFcn/* = lh_hash_fcn<_Kty> */>
class linked_hash_set
{
private:
	void assign(const linked_hash_set& rhs);

	lh_entry<value_type> _head;
	_linked_hash_set _lhs;
}; // end class linked_hash_set

_linked_hash_set 是一个STL的标准哈希表集合,每个元素是链表的节点(可是该数据结构实际并不对链表地址做哈希,而是对key做hash操作)

默认的哈希函数对象(如果自行替换哈希函数,可能会导致该数据结构失效)

	struct lhs_hasher
	{
		lhs_hasher() : cmp() {}
#if defined(_MSC_VER) && _MSC_VER < 1500
		enum {	// parameters for hash table
			bucket_size = 4,	// 0 < bucket_size
			min_buckets = 8
		};	// min_buckets = 2 ^^ N, 0 < N

		inline bool operator()(const lh_entry<_Kty>* lhs, const lh_entry<_Kty>* rhs) const {
			return cmp(lhs->val, rhs->val);
		}
#else
		inline size_t operator()(const lh_entry<_Kty>* entry) const {
			return cmp(entry->val);
		}
#endif

	private:
		HashFcn cmp;
	}; // end class lhs_hasher

然后骚操作来了。

	iterator find(const key_type& key) {
		_lhs_iter it = _lhs.find((lh_entry<value_type>*)&key);
		return iterator(it != _lhs.end() ? *it : &_head);
	}

可以使用key直接查找到链表的节点,对返回的迭代器解引用即可。

这样解决了多对象之间多次拷贝的问题。




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值