GHashTable如何使用哈希值来存储其节点?
哈希值用于确定键在 GHashTable 数据结构中的存储位置。
但是哈希值是如何使用的呢?
似乎 g_hash_table_foreach()< /code>
从
0
到 N
- 1
遍历表格N
个节点。我使用此函数打印出每个节点的哈希、键和值:
test.c
#include <glib.h>
static GHashTable* _htab = NULL;
static int VALS[] = { 1, 2, 3, 4, 5 };
struct kv {
gpointer key;
gpointer val;
};
static struct kv KEYVALS[] = {
{ "aaa", &VALS[0] },
{ "a", &VALS[1] },
{ "b", &VALS[2] },
{ "bbbbbb", &VALS[3] },
{ "aaaaa", &VALS[4] }
};
static void iter(gpointer key, gpointer val, gpointer data) {
g_printf("key=%s(%p) val=%d(%p) hash=%u\n",
(char*)key, key,
*(int*)val, val,
g_str_hash((char*)key)
);
}
int main(int argc, char* argv[]) {
int i;
if (!_htab) {
_htab = g_hash_table_new(g_str_hash, g_str_equal);
g_assert(_htab);
}
for (i = 0; i < sizeof(KEYVALS) / sizeof(KEYVALS[0]); i++) {
g_hash_table_insert(_htab, KEYVALS[i].key, KEYVALS[i].val);
}
g_hash_table_foreach(_htab, iter, NULL);
g_hash_table_remove_all(_htab);
g_hash_table_destroy(_htab);
return 0;
}
即使多次运行后,输出也始终相同(指针值除外),因此这里使用了某种算法。 节点的哈希值如何(例如,来自 g_str_hash()
) 确定它在 GHashTable
中的存储位置?
$ gcc `pkg-config --cflags glib-2.0` `pkg-config --libs glib-2.0` test.c
$ ./a.out
key=bbbbbb(0x10f142ee0) val=4(0x10f1430ac) hash=4087176913
key=a(0x10f142edc) val=2(0x10f1430a4) hash=177670
key=b(0x10f142ede) val=3(0x10f1430a8) hash=177671
key=aaaaa(0x10f142ee7) val=5(0x10f1430b0) hash=252781386
key=aaa(0x10f142ed8) val=1(0x10f1430a0) hash=193485928
$
The documentation for g_hash_table_new()
indicates
Hash values are used to determine where keys are stored within the GHashTable data structure.
but how are the hash values used?
It seems g_hash_table_foreach()
traverses the table from 0
to N
- 1
for N
nodes. I used this function to print out the hash, key, and value for each node:
test.c
#include <glib.h>
static GHashTable* _htab = NULL;
static int VALS[] = { 1, 2, 3, 4, 5 };
struct kv {
gpointer key;
gpointer val;
};
static struct kv KEYVALS[] = {
{ "aaa", &VALS[0] },
{ "a", &VALS[1] },
{ "b", &VALS[2] },
{ "bbbbbb", &VALS[3] },
{ "aaaaa", &VALS[4] }
};
static void iter(gpointer key, gpointer val, gpointer data) {
g_printf("key=%s(%p) val=%d(%p) hash=%u\n",
(char*)key, key,
*(int*)val, val,
g_str_hash((char*)key)
);
}
int main(int argc, char* argv[]) {
int i;
if (!_htab) {
_htab = g_hash_table_new(g_str_hash, g_str_equal);
g_assert(_htab);
}
for (i = 0; i < sizeof(KEYVALS) / sizeof(KEYVALS[0]); i++) {
g_hash_table_insert(_htab, KEYVALS[i].key, KEYVALS[i].val);
}
g_hash_table_foreach(_htab, iter, NULL);
g_hash_table_remove_all(_htab);
g_hash_table_destroy(_htab);
return 0;
}
The output is always the same (other than the pointer values) even after multiple runs, so there's some kind of algorithm used here. How does a node's hash (e.g., from g_str_hash()
) determine where it's stored in the GHashTable
?
$ gcc `pkg-config --cflags glib-2.0` `pkg-config --libs glib-2.0` test.c
$ ./a.out
key=bbbbbb(0x10f142ee0) val=4(0x10f1430ac) hash=4087176913
key=a(0x10f142edc) val=2(0x10f1430a4) hash=177670
key=b(0x10f142ede) val=3(0x10f1430a8) hash=177671
key=aaaaa(0x10f142ee7) val=5(0x10f1430b0) hash=252781386
key=aaa(0x10f142ed8) val=1(0x10f1430a0) hash=193485928
$
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
data:image/s3,"s3://crabby-images/d5906/d59060df4059a6cc364216c4d63ceec29ef7fe66" alt="扫码二维码加入Web技术交流群"
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(1)
让我们看看源代码可能告诉我们什么。
哈希值以某个值为模。让我们看看它可能是什么。
啊哈,来自预定义素数的
static const
数组的素数。但它设置在哪里呢?因此
,在创建 GHashTable 时,会为其分配某个素数,并用于将哈希值转换为索引。之后,当哈希表增长时,再次调用 g_hash_table_set_shift 来更新素值,并导致给定哈希值返回不同的索引。
Let's see what the source code might tell us.
The hash value is taken modulo some value. Let's see what it might be.
Aha, a prime number from a
static const
array of predefined primes. But where is it set?and
So on GHashTable's creation a certain prime number is assigned to it and it's used to translate a hash value to an index. Afterwards, when the hash table grows,
g_hash_table_set_shift
is called once again to update the prime value and causes different indices to be returned for given hash values.