Redis原理:rename命令

RENAME key newkey

将一个key重命名为新key,如果key不存在,则会返回异常。如果newKey已经存在,则会被覆盖,其实newKey会被显示的删除,所以如果newKey是一个大key,则会引起延迟。

源码

复制代码
void renameCommand(client *c)
{
    //调用rename通用方法
    //第二个参数为false
    renameGenericCommand(c, 0);
}

void renameGenericCommand(client *c, int nx)
{
    robj *o;
    long long expire;
    int samekey = 0;

    /* When source and dest key is the same, no operation is performed,
     * if the key exists, however we still return an error on unexisting key. */
    // 判断是否源key与目标key是否相同,相同则不会处理
    if (sdscmp(c->argv[1]->ptr, c->argv[2]->ptr) == 0)
        samekey = 1;
    // 如果旧key不存在,则报错
    if ((o = lookupKeyWriteOrReply(c, c->argv[1], shared.nokeyerr)) == NULL)
        return;
    // 如果前后key相同
    if (samekey)
    {
        // 如果nx == 0 则返回ok
        addReply(c, nx ? shared.czero : shared.ok);
        return;
    }
    // 增加引用计数
    incrRefCount(o);
    // 判断旧key是否过期
    expire = getExpire(c->db, c->argv[1]);
    // 如果目标key存在
    if (lookupKeyWrite(c->db, c->argv[2]) != NULL)
    {
        // nx == 1时,则需要目标key不存在
        if (nx)
        {
            //引用计数减一
            decrRefCount(o);
            addReply(c, shared.czero);
            return;
        }
        /* Overwrite: delete the old key before creating the new one
         * with the same name. */
        //目标key存在,则显示删除目标key
        dbDelete(c->db, c->argv[2]);
    }
    // 添加目标key
    dbAdd(c->db, c->argv[2], o);
    // 设置过期时间
    if (expire != -1)
        setExpire(c, c->db, c->argv[2], expire);
    // 删除旧key
    dbDelete(c->db, c->argv[1]);
    // 通知key修改
    signalModifiedKey(c, c->db, c->argv[1]);
    signalModifiedKey(c, c->db, c->argv[2]);
    // 通知事件
    notifyKeyspaceEvent(NOTIFY_GENERIC, "rename_from",
                        c->argv[1], c->db->id);
    notifyKeyspaceEvent(NOTIFY_GENERIC, "rename_to",
                        c->argv[2], c->db->id);
    server.dirty++;
    addReply(c, nx ? shared.cone : shared.ok);
}

从以上的源码也可以看出,其实做了以下几件事

  1. 判断源key是否存在,不存在则报错
  2. 判断源key与目标key是否相同,如果相同则直接返回
  3. 获取源key的值,并计数加1
  4. 获取源key的过期时间
  5. 如果nx == 1, 则目标key存在,会报错
  6. 如果nx == 0, 如果目标key存在,会先删除
  7. 将源key对应的值,设置到目标key上
  8. 删除源key
  9. 设置目标key的过期时间

从这里也可以看到,源key与目标key都会执行一次删除,如果两个都是大key,都可能会导致延迟。

相关推荐
wei_shuo21 分钟前
飞算 JavaAI 开发助手:深度学习驱动下的 Java 全链路智能开发新范式
java·开发语言·飞算javaai
爱上语文39 分钟前
Redis基础(6):SpringDataRedis
数据库·redis·后端
欧阳秦穆43 分钟前
apoc-5.24.0-extended.jar 和 apoc-4.4.0.36-all.jar 啥区别
java·jar
岁忧1 小时前
(LeetCode 面试经典 150 题 ) 58. 最后一个单词的长度 (字符串)
java·c++·算法·leetcode·面试·go
Java初学者小白1 小时前
秋招Day14 - Redis - 应用
java·数据库·redis·缓存
代码老y1 小时前
Spring Boot + 本地部署大模型实现:优化与性能提升
java·spring boot·后端
GodKeyNet1 小时前
设计模式-桥接模式
java·设计模式·桥接模式
奈斯ing1 小时前
【Redis篇】数据库架构演进中Redis缓存的技术必然性—高并发场景下穿透、击穿、雪崩的体系化解决方案
运维·redis·缓存·数据库架构
一眼万年042 小时前
Redis Cluster模式
redis·微服务
guojl2 小时前
Java多任务编排技术
java