面试突击 002 | Redis 是如何处理已过时元素的?

1 面试题面试

Redis 如何处理已过时的元素?
2 涉及知识点redis

此问题涉及如下知识点:数据库

  • 过时删除策略有哪些?
  • 这些过时策略有哪些优缺点?
  • Redis 使用的是什么过时策略?
  • Redis 是如何优化和执行过时策略的?
    3 答案

常见的过时策略:服务器

  • 定时删除
  • 惰性删除
  • 按期删除
    1)定时删除

在设置键值过时时间时,建立一个定时事件,当过时时间到达时,由事件处理器自动执行键的删除操做。
① 优势dom

保证内存能够被尽快的释放
② 缺点异步

在 Redis 高负载的状况下或有大量过时键须要同时处理时,会形成 Redis 服务器卡顿,影响主业务执行。
2)惰性删除ide

不主动删除过时键,每次从数据库获取键值时判断是否过时,若是过时则删除键值,并返回 null。
① 优势函数

由于每次访问时,才会判断过时键,因此此策略只会使用不多的系统资源。
② 缺点优化

系统占用空间删除不及时,致使空间利用率下降,形成了必定的空间浪费。
③ Redis 源码解析this

惰性删除的源码位于 src/db.c 文件的 expireIfNeeded 方法中,源码以下:

int expireIfNeeded(redisDb *db, robj *key) {
    // 判断键是否过时
    if (!keyIsExpired(db,key)) return 0;
    if (server.masterhost != NULL) return 1;
    /* 删除过时键 */
    // 增长过时键个数
    server.stat_expiredkeys++;
    // 传播键过时的消息
    propagateExpire(db,key,server.lazyfree_lazy_expire);
    notifyKeyspaceEvent(NOTIFY_EXPIRED,
        "expired",key,db->id);
    // server.lazyfree_lazy_expire 为 1 表示异步删除(懒空间释放),反之同步删除
    return server.lazyfree_lazy_expire ? dbAsyncDelete(db,key) :
                                         dbSyncDelete(db,key);
}
// 判断键是否过时
int keyIsExpired(redisDb *db, robj *key) {
    mstime_t when = getExpire(db,key);
    if (when < 0) return 0; /* No expire for this key */
    /* Don't expire anything while loading. It will be done later. */
    if (server.loading) return 0;
    mstime_t now = server.lua_caller ? server.lua_time_start : mstime();
    return now > when;
}
// 获取键的过时时间
long long getExpire(redisDb *db, robj *key) {
    dictEntry *de;
    /* No expire? return ASAP */
    if (dictSize(db->expires) == 0 ||
       (de = dictFind(db->expires,key->ptr)) == NULL) return -1;
    /* The entry was found in the expire dict, this means it should also
     * be present in the main dict (safety check). */
    serverAssertWithInfo(NULL,key,dictFind(db->dict,key->ptr) != NULL);
    return dictGetSignedIntegerVal(de);
}

全部对数据库的读写命令在执行以前,都会调用 expireIfNeeded 方法判断键值是否过时,过时则会从数据库中删除,反之则不作任何处理。
3)按期删除

每隔一段时间检查一次数据库,随机删除一些过时键。
Redis 默认每秒进行 10 次过时扫描,此配置可经过 Redis 的配置文件 redis.conf 进行配置,配置键为 hz 它的默认值是 hz 10 。
须要注意的是:Redis 每次扫描并非遍历过时字典中的全部键,而是采用随机抽取判断并删除过时键的形式执行的。
按期删除的执行流程:
面试突击 002 | Redis 是如何处理已过时元素的?
① 优势

经过限制删除操做的时长和频率,来减小删除操做对 Redis 主业务的影响,同时也能删除一部分过时的数据减小了过时键对空间的无效占用。
② 缺点

内存清理方面没有定时删除效果好,同时没有惰性删除使用的系统资源少。
③ Redis 源码解析

按期删除的核心源码在 src/expire.c 文件下的 activeExpireCycle 方法中,源码以下:

void activeExpireCycle(int type) {
    static unsigned int current_db = 0; /* 上次按期删除遍历到的数据库ID */
    static int timelimit_exit = 0;      /* Time limit hit in previous call? */
    static long long last_fast_cycle = 0; /* 上一次执行快速按期删除的时间点 */
    int j, iteration = 0;
    int dbs_per_call = CRON_DBS_PER_CALL; // 每次按期删除,遍历的数据库的数量
    long long start = ustime(), timelimit, elapsed;
    if (clientsArePaused()) return;
    if (type == ACTIVE_EXPIRE_CYCLE_FAST) {
        if (!timelimit_exit) return;
        // ACTIVE_EXPIRE_CYCLE_FAST_DURATION 是快速按期删除的执行时长
        if (start < last_fast_cycle + ACTIVE_EXPIRE_CYCLE_FAST_DURATION*2) return;
        last_fast_cycle = start;
    }
    if (dbs_per_call > server.dbnum || timelimit_exit)
        dbs_per_call = server.dbnum;
    // 慢速按期删除的执行时长
    timelimit = 1000000*ACTIVE_EXPIRE_CYCLE_SLOW_TIME_PERC/server.hz/100;
    timelimit_exit = 0;
    if (timelimit <= 0) timelimit = 1;
    if (type == ACTIVE_EXPIRE_CYCLE_FAST)
        timelimit = ACTIVE_EXPIRE_CYCLE_FAST_DURATION; /* 删除操做的执行时长 */
    long total_sampled = 0;
    long total_expired = 0;
    for (j = 0; j < dbs_per_call && timelimit_exit == 0; j++) {
        int expired;
        redisDb *db = server.db+(current_db % server.dbnum);
        current_db++;
        do {
            // .......
            expired = 0;
            ttl_sum = 0;
            ttl_samples = 0;
            // 每一个数据库中检查的键的数量
            if (num > ACTIVE_EXPIRE_CYCLE_LOOKUPS_PER_LOOP)
                num = ACTIVE_EXPIRE_CYCLE_LOOKUPS_PER_LOOP;
            // 从数据库中随机选取 num 个键进行检查
            while (num--) {
                dictEntry *de;
                long long ttl;
                if ((de = dictGetRandomKey(db->expires)) == NULL) break;
                ttl = dictGetSignedInteger
                // 过时检查,并对过时键进行删除
                if (activeExpireCycleTryExpire(db,de,now)) expired++;
                if (ttl > 0) {
                    /* We want the average TTL of keys yet not expired. */
                    ttl_sum += ttl;
                    ttl_samples++;
                }
                total_sampled++;
            }
            total_expired += expired;
            if (ttl_samples) {
                long long avg_ttl = ttl_sum/ttl_samples;
                if (db->avg_ttl == 0) db->avg_ttl = avg_ttl;
                db->avg_ttl = (db->avg_ttl/50)*49 + (avg_ttl/50);
            }
            if ((iteration & 0xf) == 0) { /* check once every 16 iterations. */
                elapsed = ustime()-start;
                if (elapsed > timelimit) {
                    timelimit_exit = 1;
                    server.stat_expired_time_cap_reached_count++;
                    break;
                }
            }
            /* 每次检查只删除 ACTIVE_EXPIRE_CYCLE_LOOKUPS_PER_LOOP/4 个过时键 */
        } while (expired > ACTIVE_EXPIRE_CYCLE_LOOKUPS_PER_LOOP/4);
    }
    // .......
}

activeExpireCycle 方法在规定的时间,分屡次遍历各个数据库,从过时字典中随机检查一部分过时键的过时时间,删除其中的过时键。
这个函数有两种执行模式,一个是快速模式一个是慢速模式,体现是代码中的 timelimit 变量,这个变量是用来约束此函数的运行时间的。快速模式下 timelimit 的值是固定的,等于预约义常量 ACTIVE_EXPIRE_CYCLE_FAST_DURATION,慢速模式下,这个变量的值是经过 1000000*ACTIVE_EXPIRE_CYCLE_SLOW_TIME_PERC/server.hz/100 计算的。
总结

本文讲了常见的过时删除策略:

  • 定时删除
  • 惰性删除
  • 按期删除 而 Redis 使用的是惰性删除 + 按期删除的组合策略。
    【END】
    近期热文

面试珍藏:最多见的200多道Java面试题(2019年最新版)
Java面试详解(2020版):500+ 面试题和核心知识点详解
面试突击 | Redis 如何从海量数据中查询出某一个 Key?视频版
关注下方二维码,订阅更多精彩内容
面试突击 002 | Redis 是如何处理已过时元素的?

相关文章
相关标签/搜索