Redis中主键失效的原理及实现机制剖析
- 行业动态
- 2024-02-19
- 2864
Redis主键失效通过EXPIRE等命令设置,到期自动删除,更新覆盖或DEL命令可撤销失效时间,RENAME命令则传递失效时间给新键。深入了解其机制有助于高效利用缓存数据。
深入剖析Redis主键失效原理及实现机制
Redis作为一款高性能的键值对存储系统,被广泛应用于缓存、消息队列、分布式锁等场景,在使用Redis的过程中,我们经常会遇到键值对失效的情况,那么Redis中主键失效的原理是什么?又是如何实现的呢?本文将深入剖析Redis主键失效的原理及实现机制。
Redis主键失效原理
在Redis中,主键失效主要分为两种情况:自然失效和手动失效。
1、自然失效
自然失效是指键值对在Redis中存储的时间超过了设定的过期时间,Redis会自动删除这些键值对,自然失效的实现依赖于Redis的过期键清理策略。
Redis的过期键清理策略主要有以下几种:
(1)惰性删除:在客户端访问键时,检查键是否已经过期,如果已经过期,则删除该键,并返回空,这种策略的优点是操作简单,缺点是内存使用效率不高,可能会出现大量过期键占用内存的情况。
(2)定时删除:Redis内部维护一个定时任务,按照一定的频率扫描数据库中的键,删除过期的键,这种策略可以有效地清理过期键,但会增加CPU的负担。
(3)定期删除:定期删除是定时删除的优化版本,它将定时扫描调整为周期性扫描,每次扫描只处理部分键,从而降低CPU的负担。
2、手动失效
手动失效是指通过DEL命令或其他相关命令手动删除键值对,这种情况下,键值对会立即失效。
Redis主键失效实现机制
下面我们将从源码角度分析Redis主键失效的实现机制。
1、自然失效实现机制
(1)惰性删除实现
在Redis中,惰性删除主要在db.c文件中的lookupKey函数中实现:
robj *lookupKey(redisDb *db, robj *key, int flags) { dictEntry *de = dictFind(db->dict, key->ptr); if (de) { robj *val = dictGetVal(de); if (expireIfNeeded(db, key) == 0) { /* If we are in the context of a Lua script, we return the * value without checking if we need to propagate the expired * key to AOF / slaves. */ if (server.lua_caller) return val; if (flags & LOOKUP_NOTOUCH) { /* This is a read-only lookup, don't touch the key */ } else { /* Update the access time for the ageing algorithm. * Don't do it if we have a saving child, as this will trigger * a copy on write madness. */ if (!hasActiveChildProcess()) updateKeyAccessTime(key); } return val; } else { /* Key expired. If we are in the context of a script, it is up to * the script to handle the key expiration. Otherwise, we return * NULL to the caller, who should handle the key expiration * properly. */ if (server.lua_caller) return NULL; } } else { /* No key */ } return NULL; }
在这个函数中,如果找到了键,会调用expireIfNeeded函数检查键是否过期,如果过期,删除键并返回0。
(2)定时删除和定期删除实现
定时删除和定期删除的实现主要在redis.c文件中的activeExpireCycle函数中:
void activeExpireCycle(int type) { /* This function has some global state in order to continue the work * incrementally across calls. */ static unsigned int current_db = 0; /* Last DB tested. */ static int timelimit_exit = 0; /* Time limit hit in previous call? */ static long long last_fast_cycle = 0; /* When last fast cycle ran. */ int j, iteration = 0; int dbs_per_call = CRON_DBS_PER_CALL; long long start = ustime(), timelimit; if (type == ACTIVE_EXPIRE_CYCLE_FAST) { if (start < last_fast_cycle + ACTIVE_EXPIRE_CYCLE_FAST_DURATION) return; last_fast_cycle = start; } /* We usually should test CRON_DBS_PER_CALL per iteration, with * two exceptions: * * 1) Don't test more DBs than we have. * 2) If last time we hit the time limit, we want to scan all DBs * in this iteration, as there is work to do in some DB and we don't want * expired keys to use memory for too much time. */ if (dbs_per_call > server.dbnum || timelimit_exit) dbs_per_call = server.dbnum; /* We can use at max ACTIVE_EXPIRE_CYCLE_SLOW_TIME_PERC percentage of CPU time * per iteration. Since this function gets called with a frequency of * server.hz times per second, the following is the max amount of time * we can spend in this function. */ timelimit = 1000000 * ACTIVE_EXPIRE_CYCLE_SLOW_TIME_PERC / server.hz / 100; /* Iterate over a few databases at a time. */ for (j = 0; j < dbs_per_call; j++) { int expired; redisDb *db = server.db+(current_db % server.dbnum); /* Increment the DB now so we are sure if we run out of time * in the current iteration we'll restart from the next DB. */ current_db++; /* Continue to expire if at the end of the cycle more than 25% * of the keys were expired. */ do { unsigned long num, slots; long long now, ttl_sum; int ttl_samples; /* If there is nothing to expire try next DB. */ if ((num = dictSize(db->dict)) == 0) { db->avg_ttl = 0; break; } /* When we have a lot of keys to expire, we get a time sample * to check later if the CPU time is too high. */ if (num > ACTIVE_EXPIRE_CYCLE_LOOKUPS_PER_LOOP) start = ustime(); /* Here we count the number of keys that are going to be * expired in this loop, and the number of keys we actually * look at to see if they should be expired. */ expired = 0; ttl_sum = 0; ttl_samples = 0; if (type == ACTIVE_EXPIRE_CYCLE_FAST) { slots = dictSlots(db->dict); /* Fast mode: Sample keys from the dictionary. */ while (slots--) { dictEntry *de = dictGetRandomKey(db->dict); long long ttl; if (dictGetVal(de) == NULL) { /*_expired++; /* Key is already logically expired. */ continue; } ttl = dictGetSignedIntegerVal(de) - now; if (activeExpireCycleTryExpire(db, de, now)) { expired++; } else { /* If the key is non expired, add its TTL to the sum. */ if (ttl > 0) { ttl_sum += ttl; ttl_samples++; } } } } else { /* Slow mode: Check every key. */ dictEntry *de = dictGetSafeIterator(db->dict); while ((de = dictNext(de)) != NULL) { long long ttl; if (dictGetVal(de) == NULL) { expired++; continue; } ttl = dictGetSignedIntegerVal(de) - now; if (activeExpireCycleTryExpire(db, de, now)) { expired++; } else { /* If the key is non expired, add its TTL to the sum. */ if (ttl > 0) { ttl_sum += ttl; ttl_samples++; } } } dictReleaseIterator(de); } /* Update the average TTL stats for this DB. */ if (ttl_samples) { long long avg_ttl = ttl_sum / ttl_samples; /* Do a simple running average with a few samples. * We just use the current estimate if the previous one * was zero, otherwise we combine the two. */ if (db->avg_ttl == 0) { db->avg_ttl = avg_ttl; } else { db->avg_ttl = (db->avg_ttl/2) + (avg_ttl/2); } } /* We can't block forever here even if there are many keys to * expire. So after a given amount of milliseconds return to the * caller waiting for the other active expire cycle. */ if (ustime() - start > timelimit) { timelimit_exit = 1; break; } } while (expired > ACTIVE_EXPIRE_CYCLE_LOOKUPS_PER_LOOP/4); } /* Update our estimate of keys existing but yet to be expired. */ updateKeyspaceEvents(); /* We don't repeat the cycle if there are less than 25% of keys to * expire in the DB we just handled, however if we exited because of the * time limit, we'll try again later
本站发布或转载的文章及图片均来自网络,其原创性以及文中表达的观点和判断不代表本站,有问题联系侵删!
本文链接:http://www.xixizhuji.com/fuzhu/214052.html