Redis的内存回收和对象共享

Redis中的内存回收

因为 C 语言并不具备自动的内存回收 功能, 所以Redis在自己的对象系统中构建了一个引用计数 技术实现的内存回收机制, 通过这一机制, 程序可以通过跟踪对象的引用计数信息 , 在适当的时候自动释放对象并进行内存回收

每个对象的引用计数 信息由 redisObject 结构的 refcount 属性记录:

c 复制代码
typedef struct redisObject {

    // ...

    // 引用计数
    int refcount;

    // ...

} robj;

对象的引用计数信息 会随着对象的使用状态而不断变化

  • 在创建一个新对象时, 引用计数的值会被初始化为 1
  • 当对象被一个新程序使用时, 它的引用计数值会被增一;
  • 当对象不再被一个程序使用时, 它的引用计数值会被减一;
  • 当对象的引用计数值变为 0 时, 对象所占用的内存会被释放。
函数 作用
incrRefCount 将对象的引用计数值增一。
decrRefCount 将对象的引用计数值减一, 当对象的引用计数值等于 0 时, 释放对象。
resetRefCount 将对象的引用计数值设置为 0 , 但并不释放对象, 这个函数通常在需要重新设置对象的引用计数值时使用。

对象的生命周期

对象的整个生命周期可以划分为创建对象操作对象释放对象三个阶段。

c 复制代码
// 创建一个字符串对象 s ,对象的引用计数为 1
robj *s = createStringObject(...)

// 对象 s 执行各种操作 ...

// 将对象 s 的引用计数减一,使得对象的引用计数变为 0
// 导致对象 s 被释放
decrRefCount(s)

对象共享

对象的引用除了进行内存回收 之外,还带有对象共享的作用。

假设键A创建一个包含整数值100的字符串作为值对象

现在假设键B也要创建一个同样的字符串作为值对象,有两种做法:

  1. 为键 B 新创建一个包含整数值 100 的字符串对象;
  2. 让键 A 和键 B 共享同一个字符串对象;

第二种方法是明显优于第一种方法的。

Redis中, 让多个键共享同一个值对象需要执行以下两个步骤:

  1. 将数据库键的值指针指向一个现有的值对象
  2. 将被共享的值对象的引用计数增一

共享对象机制对于节约内存非常有帮助, 数据库中保存的相同值对象越多, 对象共享机制就能节约越多的内存。

Redis会在初始化服务器时, 创建一万个字符串对象, 这些对象包含了从 09999 的所有整数值, 当服务器需要用到值为 09999 的字符串对象时, 服务器就会使用这些共享对象, 而不是新创建对象。

当服务器考虑将一个共享对象设置为键的值对象时, 程序需要先检查给定的共享对象和键想创建的目标对象是否完全相同 , 只有在共享对象和目标对象完全相同的情况下, 程序才会将共享对象用作键的值对象, 而一个共享对象保存的值越复杂 , 验证共享对象和目标对象是否相同所需的复杂度就会越高 , 消耗的 CPU时间也会越多

  • 如果共享对象是保存整数值 的字符串对象, 那么验证操作的复杂度为
  • 如果共享对象是保存字符串值 的字符串对象, 那么验证操作的复杂度为
  • 如果共享对象是包含了多个值(或者对象的)对象, 比如列表对象或者哈希对象, 那么验证操作的复杂度将会是

因此, 尽管共享更复杂的对象可以节约更多的内存, 但受到CPU时间的限制,Redis 只对包含整数值的字符串对象进行共享。

相关推荐
阿聪谈架构13 小时前
第08章:MCP 模型上下文协议(上)
人工智能·后端
mrchan13 小时前
markdown 画图总结
后端
开心就好202513 小时前
全面介绍iOS开发工具:Xcode、AppCode、CocoaPods、Fastlane和Git
后端·ios
uElY ITER13 小时前
基于Spring Boot 3 + Spring Security6 + JWT + Redis实现登录、token身份认证
spring boot·redis·spring
Rust研习社14 小时前
Rust Pin 解析:核心原理与异步编程实践
开发语言·后端·rust
book123_0_9914 小时前
Spring Boot 条件注解:@ConditionalOnProperty 完全解析
java·spring boot·后端
NCIN EXPE14 小时前
使用Springboot + netty 打造聊天服务(一)
java·spring boot·后端
涡能增压发动积14 小时前
日志里全是 $ref,但客户端居然没事?
后端
我叫黑大帅14 小时前
受保护的海报图片读取方案 - 在不公开静态资源目录下如何获取静态资源
后端·python·面试
我母鸡啊14 小时前
软考架构师故事系列-嵌入式技术
后端