【Redis源码】Redis命令INCR、DECR、INCRBY、DECRBY学习(二)

本文涉及的产品
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
云数据库 Tair(兼容Redis),内存型 2GB
简介: 【Redis源码】Redis命令INCR、DECR、INCRBY、DECRBY学习(二)

前言:

该篇内容为我对redis的学习记录,欢迎指正批评。文中只是放了部分调试内容,完整的可以根据流程图跟踪。

一.redis incr命令调用流程

DECR和INCR和类似只是一个是增加一个是减少。其余两个命令也大同小异;

二.源码分析

4个命令相应源码,t_string.c中:


voidincrCommand(client *c) {
   incrDecrCommand(c,1);      //incr每次增加1 命令原型:incr test1
}

voiddecrCommand(client *c) {
   incrDecrCommand(c,-1);     //incr每次减少1 命令原型:decr test1
}

voidincrbyCommand(client *c) {
   longlong incr;

   if (getLongLongFromObjectOrReply(c, c->argv[2], &incr, NULL) != C_OK) return;   //获得第二个,增加数量
   incrDecrCommand(c,incr);    //incrby每次增加输入的数 命令原型增加100:incrby test1 100
}

voiddecrbyCommand(client *c) {
   longlong incr;

   if (getLongLongFromObjectOrReply(c, c->argv[2], &incr, NULL) != C_OK) return;  //获得第二个,减少数量
   incrDecrCommand(c,-incr);   //decrby每次加少输入的数 命令原型减少100:decrby test1 100
}

incrDecrCommand函数源码 ,t_string.c中:

voidincrDecrCommand(client *c, longlong incr) {
   longlong value, oldvalue;
   robj *o, *new;

   o = lookupKeyWrite(c->db,c->argv[1]);         //查找有没有对应的键值
   if (o != NULL && checkType(c,o,OBJ_STRING)) return;      //判断类型
   if (getLongLongFromObjectOrReply(c,o,&value,NULL) != C_OK) return;  //获得键值中的数据

   oldvalue = value;
   if ((incr < 0 && oldvalue < 0 && incr < (LLONG_MIN-oldvalue)) ||
       (incr > 0 && oldvalue > 0 && incr > (LLONG_MAX-oldvalue))) {   //判断不能不能越界,超出long long范围
       addReplyError(c,"increment or decrement would overflow");
       return;
   }
   value += incr;   //与操作数相加

   /*
   判断是否大于对象共享整数,value是否在long只最小到最大之间。共享对象整数为大于10000
   */

   if (o && o->refcount == 1 && o->encoding == OBJ_ENCODING_INT &&
       (value < 0 || value >= OBJ_SHARED_INTEGERS) &&
       value >= LONG_MIN && value <= LONG_MAX)                    
   {
       new = o;
       o->ptr = (void*)((long)value);
   } else {
       new = createStringObjectFromLongLong(value); //根据数值创建一个long long的robj结构指针
       if (o) {
           dbOverwrite(c->db,c->argv[1],new);     //重写字典
       } else {
           dbAdd(c->db,c->argv[1],new);           //添加字典
       }
   }
   signalModifiedKey(c->db,c->argv[1]); //通知修改了key
   notifyKeyspaceEvent(NOTIFY_STRING,"incrby",c->argv[1],c->db->id); //发送事件通知
   server.dirty++;
   addReply(c,shared.colon);
   addReply(c,new);
   addReply(c,shared.crlf);
}


三.调试信息(看图片可能看不懂,先看流程图,根据图片再看)

下一个call断点,然后进入incrCommand

打印命令信息,以及调用栈

获得long long对象调试

增加数据部分调试

有键值是覆盖写入


相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
打赏
0
0
0
0
1
分享
相关文章
|
9天前
|
Redis如何优化频繁命令往返造成的性能瓶颈?
频繁的命令往返是Redis性能优化中需要重点关注的问题。通过使用Pipeline、Lua脚本、事务、合并命令、连接池以及合理设置网络超时,可以有效减少网络往返次数,优化Redis的性能。这些优化措施不仅提升了Redis的处理能力,还能确保系统在高并发情况下的稳定性和可靠性。
34 14
Redis是如何建立连接和处理命令的
本文主要讲述 Redis 是如何监听客户端发出的set、get等命令的。
1300 160
java连接redis和基础操作命令
通过以上内容,您可以掌握在Java中连接Redis以及进行基础操作的基本方法,进而在实际项目中灵活应用。
125 30
|
2月前
基于springboot+thymeleaf+Redis仿知乎网站问答项目源码
基于springboot+thymeleaf+Redis仿知乎网站问答项目源码
174 36
|
4月前
|
Redis命令:列表模糊删除详解
通过本文的介绍,我们详细探讨了如何在Redis中实现列表的模糊删除。虽然Redis没有直接提供模糊删除命令,但可以通过组合使用 `LRANGE`和 `LREM`命令,并在客户端代码中进行模糊匹配,来实现这一功能。希望本文能帮助你在实际应用中更有效地操作Redis列表。
168 0
Redis 命令
10月更文挑战第15天
68 0
Redis--缓存击穿、缓存穿透、缓存雪崩
缓存击穿、缓存穿透和缓存雪崩是Redis使用过程中可能遇到的常见问题。理解这些问题的成因并采取相应的解决措施,可以有效提升系统的稳定性和性能。在实际应用中,应根据具体场景,选择合适的解决方案,并持续监控和优化缓存策略,以应对不断变化的业务需求。
60 29
Redis应用—8.相关的缓存框架
本文介绍了Ehcache和Guava Cache两个缓存框架及其使用方法,以及如何自定义缓存。主要内容包括:Ehcache缓存框架、Guava Cache缓存框架、自定义缓存。总结:Ehcache适合用作本地缓存或与Redis结合使用,Guava Cache则提供了更灵活的缓存管理和更高的并发性能。自定义缓存可以根据具体需求选择不同的数据结构和引用类型来实现特定的缓存策略。
Redis应用—8.相关的缓存框架
Redis缓存设计与性能优化
Redis缓存设计与性能优化涵盖缓存穿透、击穿、雪崩及热点key重建等问题。针对缓存穿透,可采用缓存空对象或布隆过滤器;缓存击穿通过随机设置过期时间避免集中失效;缓存雪崩需确保高可用性并使用限流熔断组件;热点key重建利用互斥锁防止大量线程同时操作。此外,开发规范强调键值设计、命令使用和客户端配置优化,如避免bigkey、合理使用批量操作和连接池管理。系统内核参数如vm.swappiness、vm.overcommit_memory及文件句柄数的优化也至关重要。慢查询日志帮助监控性能瓶颈。
44 9
缓存与数据库的一致性方案,Redis与Mysql一致性方案,大厂P8的终极方案(图解+秒懂+史上最全)
缓存与数据库的一致性方案,Redis与Mysql一致性方案,大厂P8的终极方案(图解+秒懂+史上最全)