首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >为什么redis中hashmap的负载因子高达5

为什么redis中hashmap的负载因子高达5
EN

Stack Overflow用户
提问于 2020-01-03 01:23:04
回答 2查看 230关注 0票数 3

在算法类和授权书籍中,负载因子小于1,因为Java的默认值是0.75。但是在redis源代码中,负载因子是5。

代码语言:javascript
复制
54 /* Using dictEnableResize() / dictDisableResize() we make possible to
55  * enable/disable resizing of the hash table as needed. This is very important
56  * for Redis, as we use copy-on-write and don't want to move too much memory
57  * around when there is a child performing saving operations.
58  *
59  * Note that even when dict_can_resize is set to 0, not all resizes are
60  * prevented: a hash table is still allowed to grow if the ratio between
61  * the number of elements and the buckets > dict_force_resize_ratio. */
62 static int dict_can_resize = 1;
63 static unsigned int dict_force_resize_ratio = 5;

为什么会这样呢?

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2020-01-03 02:04:50

开始重新散列的负载因子是~1。dict_force_resize_ratio值是一种安全措施,因此即使禁用了重新散列,一旦达到该负载因子,它也会强制重新散列。

您可以在dict.c_dictExpandIfNeeded(dict *d)中看到这一点

代码语言:javascript
复制
/* If we reached the 1:1 ratio, and we are allowed to resize the hash
 * table (global setting) or we should avoid it but the ratio between
 * elements/buckets is over the "safe" threshold, we resize doubling
 * the number of buckets. */
if (d->ht[0].used >= d->ht[0].size &&
    (dict_can_resize ||
     d->ht[0].used/d->ht[0].size > dict_force_resize_ratio))
{
    return dictExpand(d, d->ht[0].used*2);
}

Redis允许~1开始重新散列,因为重新散列不是一下子完成的。它是通过维护两个哈希表逐步完成的。

请参阅dict.h

代码语言:javascript
复制
/* This is our hash table structure. Every dictionary has two of this as we
 * implement incremental rehashing, for the old to the new table. */
typedef struct dictht {
    dictEntry **table;
    unsigned long size;
    unsigned long sizemask;
    unsigned long used;
} dictht;

typedef struct dict {
    dictType *type;
    void *privdata;
    dictht ht[2];
    long rehashidx; /* rehashing not in progress if rehashidx == -1 */
    unsigned long iterators; /* number of iterators currently running */
} dict;

dict.c

代码语言:javascript
复制
/* Performs N steps of incremental rehashing. Returns 1 if there are still
 * keys to move from the old to the new hash table, otherwise 0 is returned.
 *
 * Note that a rehashing step consists in moving a bucket (that may have more
 * than one key as we use chaining) from the old to the new hash table, however
 * since part of the hash table may be composed of empty spaces, it is not
 * guaranteed that this function will rehash even a single bucket, since it
 * will visit at max N*10 empty buckets in total, otherwise the amount of
 * work it does would be unbound and the function may block for a long time. */
int dictRehash(dict *d, int n) {...

对于activerehashing设置,在redis.conf中还有一些额外的见解。

代码语言:javascript
复制
# Active rehashing uses 1 millisecond every 100 milliseconds of CPU time in
# order to help rehashing the main Redis hash table (the one mapping top-level
# keys to values). The hash table implementation Redis uses (see dict.c)
# performs a lazy rehashing: the more operation you run into a hash table
# that is rehashing, the more rehashing "steps" are performed, so if the
# server is idle the rehashing is never complete and some more memory is used
# by the hash table.
#
# The default is to use this millisecond 10 times every second in order to
# actively rehash the main dictionaries, freeing memory when possible.
#
# If unsure:
# use "activerehashing no" if you have hard latency requirements and it is
# not a good thing in your environment that Redis can reply from time to time
# to queries with 2 milliseconds delay.
#
# use "activerehashing yes" if you don't have such hard requirements but
# want to free memory asap when possible.
activerehashing yes
票数 4
EN

Stack Overflow用户

发布于 2020-12-03 16:53:54

因为Redis中load_factor的计算公式是:ht[0].used/d->ht[0].sizeused表示哈希表中元素的个数,size表示底层数组的大小,所以如果发生哈希冲突,元素会存储在链表中,链表的大小不包含在ht[0].size中。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/59567447

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档