hashtable
哈希表又叫散列表,是实现字典操作的中有效数据结构。通常来说,一个 hash table 包含了一个数据,其中的数据通过 index 来访问。 而 hash table 的基本原理就是通过 hash 函数建立起所有可能的 index 与其对应的位置的联系。一个 hash 函数接收一个 key,返回其 hash code, key 的类型是可变的,而 hash code 是一个整型。
由于计算一个 hash 值和通过 index 访问一个数据都是常量级的时间复杂度,所以我们可以通过这中特性实现常量级时间复杂度的查找。 如果一个 hash 函数能够保证不会有两个不同的 key 生成相同的 hash 值,那么这样的 hash table 就被称为是直接定址。然而,这只是一想法而已, 实际上这种 hash table 在现实中却是不常用的。
Chained Hash Table
链式 hash 表从本质上来讲,就是一个存放了一组链表的数组。每个链表可以看做是一个槽,我们把元素通过 hash 函数找到一个 hash 值,然后把元素的值 放入到数组中与改 hash 值对应的槽中。
Collision Resolution
当有两个 key 被 hash 到了同一个位置,就会产生冲突。链式 hash 表有一种简单的冲突解决办法:当冲突产生时,元素被简单的放在同一个槽里。这样做可能带来的问题就是, 如果在同一个位置上出现很多冲突,这个槽就会变得越来越长,这样当我们访问这个槽中的元素的时候,所花的时间也就会越来越长。
在理想状态下,我们希望所有的槽能够以同样的速度增长,这样他们就可以尽可能地保持小的容量和相同的大小。换句话说,我们的目标就是尽可能均匀分散所有的元素。这种情况 在理论上称为均匀散列。而实际中,我们只能尽可能近似的到达这种状态。
如果插入表中的元素数量远大于表中槽的数量,那么即使在均匀分布的情况下,表中的槽也会变得越来越深,表的性能也会随之迅速降低。因此我们必须要特别注意一个 hash 表的负载因子, 其定义为:
$$ \alpha = n / m $$
其中$n$是表中元素的个数,$m$是槽的数量。链式 hash 表的负载因子表示在均匀散列的情况下,表中每一个槽的最大期望存放的元素个数。 例如:有一个 hash 表,$m = 1699$, $n = 3198$,那么这个 hash 表的负载因子为$\alpha = 3198/1699 = 2$。因此在这种情况下,我们说 该 hash 表中每一个槽中的元素数量最大不要超过 2。
选择 hash 函数
一个好的 hash 函数的目标是尽可能近似于均匀散列。定义一个 hash 函数$f$,它将键$k$映射到 hash 表中的位置$x$,$x$称为$k$的 hash code:
$$ h(k) = x $$
Division Method
有一个整数键$k$,有一种最简单的将$k$映射到 hash 表中槽位的方法就是取余数:
$$ h(h) = k,mod,m $$
Multiplication Method
与求余法不同的一种方法是乘法,它将整型键$k$乘以一个常数$A (0 < A < 1)$,取所乘结果的小数部分,然后再乘以$m$,取结果的整数部分。通常情况下,$A$取$0.618$, 它由$\sqrt{5}$减$1$再除以$2$得到:
$$ h(k) = \lfloor m (kA,mod,1) \rfloor,其中 A \approx (\sqrt{5} - 1) / 2 $$
链式 hash table 的实现
基于以上理论,实现一个 hash 表已经显得非常简单了。下面直接给出代码:
1#include <stdlib.h>
2#include <stdio.h>
3#include <limits.h>
4#include <string.h>
5
6#define linger_free(ptr) if (ptr) free(ptr)
7
8typedef struct entry_s {
9 char *key;
10 char *value;
11 struct entry_s *next;
12} entry_t;
13
14typedef struct hashtable_s {
15 int size;
16 entry_t **table;
17} hashtable_t;
18
19hashtable_t *ht_create(int size)
20{
21 hashtable_t *hashtable = NULL;
22 int i;
23
24 if (size < 1) {
25 return NULL;
26 }
27
28 if ((hashtable = malloc(sizeof(hashtable_t))) == NULL) {
29 return NULL;
30 }
31
32 if ((hashtable->table = malloc(sizeof(entry_t *) * size)) == NULL) {
33 linger_free(hashtable);
34 return NULL;
35 }
36
37 for (i = 0; i < size; i++) {
38 hashtable->table[i] = NULL;
39 }
40
41 hashtable->size = size;
42 return hashtable;
43}
44
45int ht_hash(hashtable_t *hashtable, char *key)
46{
47 const char *ptr;
48 unsigned int val;
49 val = 0;
50 ptr = key;
51 while (*ptr != '\0') {
52 unsigned int tmp;
53 val = (val << 4) + (*ptr);
54 if (tmp = (val & 0xf0000000)) {
55 val = val ^ (tmp >> 24);
56 val = val ^ tmp;
57 }
58 ptr++;
59 }
60 return val % hashtable->size;
61}
62
63entry_t *ht_newpair(char *key, char *value)
64{
65 entry_t *newpair;
66 if ((newpair = malloc(sizeof(entry_t))) == NULL) {
67 return NULL;
68 }
69
70 if ((newpair->key = strdup(key)) == NULL) {
71 return NULL;
72 }
73
74 if ((newpair->value = strdup(value)) == NULL) {
75 return NULL;
76 }
77 newpair->next = NULL;
78
79 return newpair;
80}
81
82void ht_set(hashtable_t *hashtable, char *key, char *value)
83{
84 int bin = 0;
85 entry_t *newpair = NULL;
86 entry_t *next = NULL;
87 entry_t *last = NULL;
88
89 bin = ht_hash(hashtable, key);
90
91 next = hashtable->table[bin];
92
93 while (next != NULL && next->key != NULL && strcmp(key, next->key) > 0) {
94 last = next;
95 next = next->next;
96 }
97
98 if (next != NULL && next->key != NULL && strcmp(key, next->key) == 0) {
99 linger_free(next->value);
100 next->value = strdup(value);
101 } else {
102 newpair = ht_newpair(key, value);
103 if (next == hashtable->table[bin]) {
104 newpair->next = next;
105 hashtable->table[bin] = newpair;
106 } else if (next == NULL) {
107 last->next = newpair;
108 } else {
109 newpair->next = next;
110 last->next = newpair;
111 }
112 }
113}
114
115char *ht_get(hashtable_t *hashtable, char *key)
116{
117 int bin = 0;
118 entry_t *pair;
119 bin = ht_hash(hashtable, key);
120 pair = hashtable->table[bin];
121 while (pair != NULL && pair->key != NULL && strcmp(key, pair->key) > 0) {
122 pair = pair->next;
123 }
124
125 if (pair == NULL || pair->key == NULL || strcmp(key, pair->key) != 0) {
126 return NULL;
127 }
128
129 return pair->value;
130}
131
132void ht_destroy(hashtable_t *hashtable)
133{
134 entry_t *curr, *next;
135 if (hashtable->size > 0) {
136 for (int i = 0; i < hashtable->size; i++) {
137 if (hashtable->table[i] == NULL) {
138 continue;
139 }
140 curr = hashtable->table[i];
141 while (curr != NULL) {
142 next = curr->next;
143 linger_free(curr->key);
144 linger_free(curr->value);
145 linger_free(curr);
146 curr = next;
147 }
148
149 }
150 }
151 linger_free(hashtable);
152}
至此我们就实现了一个简单的链式 hash table。
给 php 扩展一个 hash table
接下来我们把这个 hash table 写成 php 扩展。由于太简单,这里就不再详细讲述实现过程,源代码可以访问https://github.com/iliubang/php_hashtable_extension.git
写了个脚本测试下性能:
1<?php
2$ht = new linger\Hashtable(65535);
3
4$n = 10000;
5
6$start = microtime(true);
7for ($i = 0; $i < $n; $i++) {
8 $ht->set("hello{$i}", "world{$i}");
9}
10
11for ($i = 0; $i < $n; $i++) {
12 $ht->get("hello{$i}");
13}
14
15echo "HashTable:" . (microtime(true) - $start), PHP_EOL;
16
17$arr = [];
18
19$start = microtime(true);
20for ($i = 0; $i < $n; $i++) {
21 $arr["hello{$i}"] = "world{$i}";
22}
23
24for ($i = 0; $i < $n; $i++) {
25 $tmp = $arr["hello{$i}"];
26}
27echo "Array:" . (microtime(true) - $start), PHP_EOL;
测试结果还是很不错的:
评论