【原创】为什么 Redis 重启后没有正确恢复之前的内存数据

本文涉及的产品
云数据库 Tair(兼容Redis),内存型 2GB
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
简介:
安装 Redis 后,默认配置下启动会得到如下日志:  
?
1
2
[3077] 05 Sep 10:01:54.181 # WARNING overcommit_memory is set to 0! Background save may fail under low memory condition.
To fix this issue add 'vm.overcommit_memory = 1' to /etc/sysctl .conf and then reboot or run the command 'sysctl vm.overcommit_memory=1' for this to take effect.
      该输出信息大致的意思是:   overcommit_memory 值目前设置为 0 !在可用内存比较少时,后台执行的 save 命令可能会失败。为了修复该问题,需要添加 'vm.overcommit_memory  = 1' 到 /etc/sysctl.conf 文件中,并   通过   重启系统或执行 'cysctl vm.overcommit_memory=1' 命令使得该配置生效。  

查看系统中的配置如下:  
?
1
2
[root@Betty ~] # sysctl vm.overcommit_memory
vm.overcommit_memory = 0
恩,确实系统配置为 0 。那么 overcommit_memory 该值的含义是什么?可以设置哪些值呢?  

网上相关文章对该参数有如下说明:  

  CommitLimit 和 Committed_As    
?
1
2
3
4
[root@Betty ~] # grep -i commit /proc/meminfo
CommitLimit:     5856624 kB
Committed_AS:     863300 kB
[root@Betty ~] #
在内存信息中可以看到 CommitLimit 和 Committed_As 两个参数。其中  
  • CommitLimit 是内存分配的上限。CommitLimit = 物理内存 * overcommit_ratio/100 + swap
  • Committed_As 是已经分配的内存大小。
  overcommit_memory    

说法一:  
overcommit_memory 参数就是控制内存分配是否可以超过 CommitLimit 值的,其指定了内核针对内存分配的策略,默认是 0 。  
  • 0 表示尽量减少 swap 的使用;
  • 1 表示允许超过 CommitLimit ;
  • 2 表示不允许超过 CommitLimit 。

说法二:  
  • 0, 表示内核将检查是否有足够的可用内存供应用进程使用;如果有足够的可用内存,则允许内存申请;否则,内存申请失败,并把错误返回给应用进程。
  • 1, 表示内核允许分配所有的物理内存,而不管当前的内存状态如何。
  • 2, 表示内核允许分配超过所有物理内存和交换空间总和的内存

说法三:  
vm.overcommit_memory 表示内核在分配内存时候做检查的方式。   这个变量可以取到 0、1、2 三个值。针对取不同的值时的处理方式,定义在内核源码 mm/mmap.c 的 __vm_enough_memory 函数中。  

取 1 时   此时宏为 OVERCOMMIT_ALWAYS,函数直接 return 0,分配成功。  
取 2 时:    此时宏为 OVERCOMMIT_NEVER,内核计算:  
?
1
内存总量*vm.overcommit_ratio /100 + SWAP 的总量
如果申请空间超过此数值,则分配失败。vm.overcommit_ratio 的默认值为 50。  
取 0 时   此时宏为 OVERCOMMIT_GUESS,内核计算:  
?
1
NR_FILE_PAGES 总量 + SWAP总量 + slab 中可以释放的内存总量
如果申请空间超过此数值,则将此数值与空闲内存总量减掉 total reserve_pages(?) 的总量相加。如果申请空间依然超过此数值,则分配失败。  

以上为粗略描述,在实际计算时,   如果非 root 进程,则在计算时候会保留 3% 的空间,而 root 进程则没有该限制    

说法四:  
vm.overcommit_memory    默认值为 0 。   从内核文档里得知,该参数有三个值,分别是:  
  • 0:当用户空间请求更多的的内存时,内核尝试估算出剩余可用的内存;
  • 1:当设这个参数值为 1 时,内核允许超量使用内存直到用完为止,主要用于科学计算;
  • 2:当设这个参数值为 2 时,内核会使用一个决不过量使用内存的算法,即系统整个内存地址空间不能超过 swap + 50% 的 RAM 值50% 参数的设定是在 overcommit_ratio 中设定。
vm.overcommit_ratio    默认值为 50 。   这个参数值只有在 vm.overcommit_memory=2 的情况下,这个参数才会生效。  

说法五:  
Controls overcommit of system memory, possibly allowing processes to allocate (but not use) more memory than is actually available.  
  • 0 - Heuristic overcommit handling. Obvious overcommits of address space are refused. Used for a typical system. It ensures a seriously wild allocation fails while allowing overcommit to reduce swap usage. root is allowed to allocate slighly more memory in this mode. This is the default.
  • 1 - Always overcommit. Appropriate for some scientific applications.
  • 2 - Don't overcommit. The total address space commit for the system is not permitted to exceed swap plus a configurable percentage (default is 50) of physical RAM. Depending on the percentage you use, in most situations this means a process will not be killed while attempting to use already-allocated memory but will receive errors on memory allocation as appropriate.

说法六(可以认为是上面英文比较准确的翻译):  
  • 0:默认值,由 Linux 内核通过一些启发式算法来决定是否超售和超售的大小,一般允许轻微的超售,拒绝一些明显不可能提供的请求,同时做一些规则限制,比如不同用户 overcommit 的大小也不一样。 
  • 1:允许,不做限制的超售,当然这个也不是无限大,还受到寻址空间的限制,32 位系统最大可能只有 4G,64 位系统大概 16T 左右。 
  • 2:禁止,禁止超售,系统能够分配的内存不会超过 swap + 实际物理内存 * overcommit_ratio,该值可以通过 /proc/sys/vm/overcommit_ratio设置,默认 50% 。


说了半天,这个和 Redis 上面给出的输出信息有什么关系呢?  
         目前我认为比较正确的解释是这样的:Redis 在执行 bgsave 的时候需要 fork 出一个子进程,一般情况下我们认为子进程所需内存空间的量等于父进程内存空间使用量,但由于 Linux 中 COW 的原因,子进程在刚刚运行时可能并不需要占用等同于父进程使用的内存量。但是尽管如此,若系统当前可用内存量较少,且在 overcommit_memory = 0 的情况下,就可能会出现由于判定内存不足导致分配失败的情况(虽然实际上可能并不真正需要那么多内存)。所以 overcommit = 1 这种设置对于 Redis 这种缓存系统显得更为有意义。  
      关于    overcommit 本身,其实还有很多可以研究的地方,以上只是本人粗浅的理解,后续将继续学习相关知识。  

=== 我是华丽的分隔线 ===  

      什么是 overcommit?overcommit 中文翻译为“超售”。   在某些操作系统中(如 Solaris),malloc 函数返回为 null 则表示系统没有可分配的内存空间。但 Linux 的内存分配采取的是一种更加积极的分配策略,它假设应用申请了内存空间后并不会立即去使用它,所以允许一定量的超售,当应用真的需要使用它的时候,操作系统可能已经通过回收了其他应用的内存空间而变得有能力去满足这个应用的需求,简单的说,就是允许应用申请比实际可分配空间(包括物理内存和 Swap)更多的内存,这个特性称为 OverCommit 。  
相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore     ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库 ECS 实例和一台目标数据库 RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
相关文章
|
1月前
|
缓存 NoSQL Redis
Redis经典问题:数据并发竞争
数据并发竞争是大流量系统(如火车票系统、微博平台)中常见的问题,可能导致用户体验下降甚至系统崩溃。本文介绍了两种解决方案:1) 加写回操作加互斥锁,查询失败快速返回默认值;2) 保持多个缓存备份,减少并发竞争概率。通过实践案例展示,成功提高了系统的稳定性和性能。
|
1月前
|
缓存 监控 NoSQL
Redis经典问题:数据不一致
在使用Redis时,缓存与数据库数据不一致会导致应用异常。主要原因包括缓存更新失败、Rehash异常等。解决方案有:重试机制、缩短缓存时间、优化写入策略、建立监控报警、定期验证一致性、采用缓存分层及数据回滚恢复机制。这些措施可确保数据最终一致性,提升应用稳定性和性能。
|
1月前
|
NoSQL 算法 Redis
redis内存淘汰策略
Redis支持8种内存淘汰策略,包括noeviction、volatile-ttl、allkeys-random、volatile-random、allkeys-lru、volatile-lru、allkeys-lfu和volatile-lfu。这些策略分别针对所有键或仅设置TTL的键,采用随机、LRU(最近最久未使用)或LFU(最少频率使用)等算法进行淘汰。
52 5
|
2月前
|
存储 编译器 数据处理
C 语言结构体与位域:高效数据组织与内存优化
C语言中的结构体与位域是实现高效数据组织和内存优化的重要工具。结构体允许将不同类型的数据组合成一个整体,而位域则进一步允许对结构体成员的位进行精细控制,以节省内存空间。两者结合使用,可在嵌入式系统等资源受限环境中发挥巨大作用。
96 11
|
2月前
|
缓存 NoSQL 关系型数据库
Redis和Mysql如何保证数据⼀致?
在项目中,为了解决Redis与Mysql的数据一致性问题,我们采用了多种策略:对于低一致性要求的数据,不做特别处理;时效性数据通过设置缓存过期时间来减少不一致风险;高一致性但时效性要求不高的数据,利用MQ异步同步确保最终一致性;而对一致性和时效性都有高要求的数据,则采用分布式事务(如Seata TCC模式)来保障。
79 14
|
2月前
|
存储 NoSQL 算法
Redis分片集群中数据是怎么存储和读取的 ?
Redis集群采用哈希槽分区算法,共有16384个哈希槽,每个槽分配到不同的Redis节点上。数据操作时,通过CRC16算法对key计算并取模,确定其所属的槽和对应的节点,从而实现高效的数据存取。
72 13
|
2月前
|
缓存 NoSQL PHP
Redis作为PHP缓存解决方案的优势、实现方式及注意事项。Redis凭借其高性能、丰富的数据结构、数据持久化和分布式支持等特点,在提升应用响应速度和处理能力方面表现突出
本文深入探讨了Redis作为PHP缓存解决方案的优势、实现方式及注意事项。Redis凭借其高性能、丰富的数据结构、数据持久化和分布式支持等特点,在提升应用响应速度和处理能力方面表现突出。文章还介绍了Redis在页面缓存、数据缓存和会话缓存等应用场景中的使用,并强调了缓存数据一致性、过期时间设置、容量控制和安全问题的重要性。
59 5
|
存储 缓存 NoSQL
Redis 漫谈 —— 分布式布隆过滤及内存使用问题分析
Redis 分布式布隆过滤及内存使用问题分析
1184 0
Redis 漫谈 —— 分布式布隆过滤及内存使用问题分析
|
1月前
|
存储 缓存 NoSQL
解决Redis缓存数据类型丢失问题
解决Redis缓存数据类型丢失问题
178 85
|
11天前
|
存储 缓存 NoSQL
云端问道21期方案教学-应对高并发,利用云数据库 Tair(兼容 Redis®*)缓存实现极速响应
云端问道21期方案教学-应对高并发,利用云数据库 Tair(兼容 Redis®*)缓存实现极速响应