Redis内存回收和对象共享的实现示例

 更新时间:2026年03月31日 09:05:34   作者:Nyarlathotep0113  
本文主要介绍了Redis内存回收和对象共享的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

Redis中的内存回收

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

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

typedef struct redisObject {
    // ...
    // 引用计数
    int refcount;
    // ...
} robj;

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

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

对象的生命周期

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

// 创建一个字符串对象 s ,对象的引用计数为 1
robj *s = createStringObject(...)
// 对象 s 执行各种操作 ...
// 将对象 s 的引用计数减一,使得对象的引用计数变为 0
// 导致对象 s 被释放
decrRefCount(s)

对象共享

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

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

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

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

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

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

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

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

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

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

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

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

到此这篇关于Redis内存回收和对象共享的实现示例的文章就介绍到这了,更多相关Redis 内存回收和对象共享内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Redis实现信息已读未读状态提示

    Redis实现信息已读未读状态提示

    这篇文章主要介绍了Redis实现信息已读未读状态提示的相关资料,需要的朋友可以参考下
    2016-04-04
  • 基于redis实现分布式锁的原理与方法

    基于redis实现分布式锁的原理与方法

    这篇文章主要给大家介绍了基于redis实现分布式锁的原理与方法,文中通过示例代码介绍的非常详细,对大家学习或者使用redis具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2020-06-06
  • redis内存空间效率问题的深入探究

    redis内存空间效率问题的深入探究

    redis缓存固然高效,可是它会占用我们系统中宝贵的内存资源,那该如何解决呢?这篇文章主要给大家介绍了关于redis内存空间效率问题的相关资料,需要的朋友可以参考下
    2021-05-05
  • Redis利用互斥锁解决缓存击穿问题

    Redis利用互斥锁解决缓存击穿问题

    使用互斥锁可以有效防止缓存击穿的情况发生,它能够保证在缓存失效时,只有一个线程或者进程能够去加载数据,其余的请求都会等待这个加载过程完成,虽然这种方式会牺牲一部分性能,但它大大提高了系统的稳定性和可用性
    2024-08-08
  • 基于Redis的分布式锁及原子性问题(短视频开发)

    基于Redis的分布式锁及原子性问题(短视频开发)

    短视频开发中,Redis分布式锁通过SETNX实现加锁与解锁,需设置超时时间避免死锁,为防止误删,释放锁时需判断线程身份,并用Lua脚本保证原子性,确保安全操作,本文给大家介绍基于Redis的分布式锁及原子性问题,感兴趣的朋友一起看看吧
    2025-06-06
  • 如何用redis setNX命令来加锁

    如何用redis setNX命令来加锁

    这篇文章主要介绍了如何用redis setNX命令来加锁,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-01-01
  • Redis如何实现投票功能

    Redis如何实现投票功能

    这篇文章主要介绍了Redis如何实现投票功能,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-05-05
  • 使用RedisAtomicInteger计数出现少计问题及解决

    使用RedisAtomicInteger计数出现少计问题及解决

    这篇文章主要介绍了使用RedisAtomicInteger计数出现少计问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-11-11
  • php结合redis实现高并发下的抢购、秒杀功能的实例

    php结合redis实现高并发下的抢购、秒杀功能的实例

    下面小编就为大家带来一篇php结合redis实现高并发下的抢购、秒杀功能的实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-12-12
  • Redis高可用之持久化

    Redis高可用之持久化

    在web服务器中,高可用是指服务器可以正常访问的时间,衡量的标准是在多长时间内可以提供正常服务(99.9%、99.99%、99.999%等等),Redis中,实现高可用的技术主要包括持久化、主从复制、哨兵和cluster集群,感兴趣的同学可以阅读本文
    2023-04-04

最新评论