`

Redis内存使用优化与存储

 
阅读更多

传统MySQL+ Memcached架构遇到的问题

实际MySQL是适合进行海量数据存储的,通过Memcached将热点数据加载到cache,加速访问,很多公司都曾经使用过这样的架构,但随着业务数据量的不断增加,和访问量的持续增长,我们遇到了很多问题:

  1. MySQL需要不断进行拆库拆表,Memcached也需不断跟着扩容,扩容和维护工作占据大量开发时间。
  2. Memcached与MySQL数据库数据一致性问题。
  3. Memcached数据命中率低或down机,大量访问直接穿透到DB,MySQL无法支撑。
  4. 跨机房cache同步问题。

众多NoSQL百花齐放,如何选择

最近几年,业界不断涌现出很多各种各样的NoSQL产品,那么如何才能正确地使用好这些产品,最大化地发挥其长处,是我们需要深入研究和思考的问题,实际归根结底最重要的是了解这些产品的定位,并且了解到每款产品的tradeoffs,在实际应用中做到扬长避短,总体上这些NoSQL主要用于解决以下几种问题

  1. 少量数据存储,高速读写访问。此类产品通过数据全部in-momery 的方式来保证高速访问,同时提供数据落地的功能,实际这正是Redis最主要的适用场景。
  2. 海量数据存储,分布式系统支持,数据一致性保证,方便的集群节点添加/删除。
  3. 这方面最具代表性的是dynamo和bigtable 2篇论文所阐述的思路。前者是一个完全无中心的设计,节点之间通过gossip方式传递集群信息,数据保证最终一致性,后者是一个中心化的方案设计,通过类似一个分布式锁服务来保证强一致性,数据写入先写内存和redo log,然后定期compat归并到磁盘上,将随机写优化为顺序写,提高写入性能。
  4. Schema free,auto-sharding等。比如目前常见的一些文档数据库都是支持schema-free的,直接存储json格式数据,并且支持auto-sharding等功能,比如mongodb。

面对这些不同类型的NoSQL产品,我们需要根据我们的业务场景选择最合适的产品。

 

前面已经分析过,Redis最适合所有数据in-momory的场景,虽然Redis也提供持久化功能,但实际更多的是一个disk-backed的功能,跟传统意义上的持久化有比较大的差别,那么可能大家就会有疑问,似乎Redis更像一个加强版的Memcached,那么何时使用Memcached,何时使用Redis呢?

Redis与Memcached的比较

    1. 网络IO模型

Memcached是多线程,非阻塞IO复用的网络模型,分为监听主线程和worker子线程,监听线程监听网络连接,接受请求后,将连接描述字pipe 传递给worker线程,进行读写IO, 网络层使用libevent封装的事件库,多线程模型可以发挥多核作用,但是引入了cache coherency和锁的问题,比如,Memcached最常用的stats 命令,实际Memcached所有操作都要对这个全局变量加锁,进行计数等工作,带来了性能损耗。

 

(Memcached网络IO模型)

Redis使用单线程的IO复用模型,自己封装了一个简单的AeEvent事件处理框架,主要实现了epoll、kqueue和select,对于单纯只有IO操作来说,单线程可以将速度优势发挥到最大,但是Redis也提供了一些简单的计算功能,比如排序、聚合等,对于这些操作,单线程模型实际会严重影响整体吞吐量,CPU计算过程中,整个IO调度都是被阻塞住的。

    1. 内存管理方面

Memcached使用预分配的内存池的方式,使用slab和大小不同的chunk来管理内存,Item根据大小选择合适的chunk存储,内存池的方式可以省去申请/释放内存的开销,并且能减小内存碎片产生,但这种方式也会带来一定程度上的空间浪费,并且在内存仍然有很大空间时,新的数据也可能会被剔除,原因可以参考Timyang的文章:http://timyang.net/data/Memcached-lru-evictions/

Redis使用现场申请内存的方式来存储数据,并且很少使用free-list等方式来优化内存分配,会在一定程度上存在内存碎片,Redis跟据存储命令参数,会把带过期时间的数据单独存放在一起,并把它们称为临时数据,非临时数据是永远不会被剔除的,即便物理内存不够,导致swap也不会剔除任何非临时数据(但会尝试剔除部分临时数据),这点上Redis更适合作为存储而不是cache。

    1. 数据一致性问题

Memcached提供了cas命令,可以保证多个并发访问操作同一份数据的一致性问题。 Redis没有提供cas 命令,并不能保证这点,不过Redis提供了事务的功能,可以保证一串 命令的原子性,中间不会被任何操作打断。

    1. 存储方式及其它方面

Memcached基本只支持简单的key-value存储,不支持枚举,不支持持久化和复制等功能

Redis除key/value之外,还支持list,set,sorted set,hash等众多数据结构,提供了KEYS

进行枚举操作,但不能在线上使用,如果需要枚举线上数据,Redis提供了工具可以直接扫描其dump文件,枚举出所有数据,Redis还同时提供了持久化和复制等功能。

    1. 关于不同语言的客户端支持

在不同语言的客户端方面,Memcached和Redis都有丰富的第三方客户端可供选择,不过因为Memcached发展的时间更久一些,目前看在客户端支持方面,Memcached的很多客户端更加成熟稳定,而Redis由于其协议本身就比Memcached复杂,加上作者不断增加新的功能等,对应第三方客户端跟进速度可能会赶不上,有时可能需要自己在第三方客户端基础上做些修改才能更好的使用。

根据以上比较不难看出,当我们不希望数据被踢出,或者需要除key/value之外的更多数据类型时,或者需要落地功能时,使用Redis比使用Memcached更合适。

关于Redis的一些周边功能

Redis除了作为存储之外还提供了一些其它方面的功能,比如聚合计算、pubsub、scripting等,对于此类功能需要了解其实现原理,清楚地了解到它的局限性后,才能正确的使用,比如pubsub功能,这个实际是没有任何持久化支持的,消费方连接闪断或重连之间过来的消息是会全部丢失的,又比如聚合计算和scripting等功能受Redis单线程模型所限,是不可能达到很高的吞吐量的,需要谨慎使用。

总的来说Redis作者是一位非常勤奋的开发者,可以经常看到作者在尝试着各种不同的新鲜想法和思路,针对这些方面的功能就要求我们需要深入了解后再使用。

总结:

  1. Redis使用最佳方式是全部数据in-memory。
  2. Redis更多场景是作为Memcached的替代者来使用。
  3. 当需要除key/value之外的更多数据类型支持时,使用Redis更合适。
  4. 当存储的数据不能被剔除时,使用Redis更合适。

后续关于Redis文章计划:

  1. Redis数据类型与容量规划。
  2. 如何根据业务场景搭建稳定,可靠,可扩展的Redis集群。
  3. Redis参数,代码优化及二次开发基础实践。

 

分享到:
评论

相关推荐

    Redis内存使用优化与存储中文最新版本

    本文将从多个角度来探讨Redis内存使用优化与存储的最新策略,为读者提供实用的技术指导。 首先,优化Redis内存使用的关键在于理解其内部数据结构及其内存分配机制。Redis支持多种数据类型,包括字符串(Strings)、...

    redis内存模型介绍

    通过以上介绍,我们可以了解到Redis内存模型的核心概念,包括数据类型的定义、持久化机制的选择以及如何利用虚拟内存机制和主从复制机制来优化Redis的应用性能。这些知识点对于深入理解Redis的工作原理和高效使用...

    Go 实现的 Redis 内存分析工具

    Redis 是一个高性能的键值存储系统,常用于缓存和数据持久化,而这款工具能够帮助用户更有效地管理和优化 Redis 的内存使用情况。 描述中提到,此工具可以用来找出占用内存最多的 Key,并将分析结果导出为 CSV 文件...

    通过Key前缀分析Redis的内存占用按内存大小排序导出结果到csv文件

    总结来说,通过Key前缀分析Redis内存占用并导出结果到CSV文件,是一个涉及Redis内存管理、PHP编程、数据遍历、排序算法和文件操作等多个IT领域的综合实践。了解并掌握这些知识点,对于优化Redis的内存使用,提升系统...

    Redis硬核性能优化

    6. 避免Redis内存发生Swap:配置禁用swap,确保数据始终在物理内存中。 7. 内存大页:根据实际需求决定是否启用,避免不必要的性能损失。 8. 使用Lazy Free:延迟删除,减少对主线程的影响,如`UNLINK`命令。 9. ...

    redis内存存储结构分析

    #### 三、Redis内存优化策略 针对Redis内存占用问题,可以采取以下几种策略进行优化: 1. **数据分层**:通过将热点数据与冷数据分开存储,只将热点数据放在内存中。 2. **数据压缩**:对于大文本数据,可以考虑在...

    美团在Redis上踩过的一些坑-3.redis内存占用飙升

    在使用Redis的过程中,他们遇到了一些问题,特别是关于Redis内存占用飙升的问题。下面我们将深入探讨这个问题以及可能的解决方案。 Redis内存占用飙升的原因多种多样,可能是由于以下几点: 1. **数据结构不当**:...

    redis内存数据库2.4.1

    综上所述,Redis 2.4.1作为一个内存数据库,通过其内存存储、持久化、多数据结构、发布订阅、主从复制以及丰富的API支持,成为了一个功能强大、性能优秀的数据存储解决方案。尽管随着时间的推移,Redis已经发展到了...

    Redis内存模型.docx

    理解Redis内存模型对于优化内存使用、估算内存需求和解决问题至关重要。例如,通过监控`used_memory_rss`和`used_memory`的比率,我们可以评估内存碎片的情况。当比率过高时,可能需要考虑采取措施减少碎片,或者...

    redis_3.2.9_内存分布分析

    Redis使用这些数据结构来优化存储效率和提升访问速度。例如,SDS用于存储字符串,而字典则用于存储键值对映射,两者都充分利用了内存的空间效率。而整数集合和压缩列表则用于处理键值对中的整数集合和小的对象列表,...

    查看Redis内存信息的命令

    首先,`info memory`是查看Redis内存使用情况的基础命令。执行`redis-cli info memory`会返回一系列内存统计数据,其中包括: 1. `used_memory`: 表示Redis分配器分配的总内存大小,即存储所有数据占用的内存。 2. ...

    Redis内存过期策略

    ### Redis内存过期策略详解 #### 一、引言 Redis作为一款高性能的键值存储系统,在实际应用中经常被用作缓存解决方案。为了确保内存的有效利用与系统的稳定性,Redis设计了一套高效的内存管理机制,其中包括了针对...

    redis集群, tomcat优化以及 MySQL5.6优化

    5. 集群配置与管理:使用 `redis-cli` 或专门的工具如 `redis-trib.rb` 进行集群的创建、检查和调整。 Tomcat 优化策略: Tomcat 是一个广泛应用的 Java Servlet 容器,针对其性能优化主要集中在以下几个方面: 1....

    Redis内存碎片:深度解析与优化策略

    1. **内存中数据存储**:Redis 将所有数据保持在内存中,这使得读写速度非常快。 2. **持久化**:尽管 Redis 是一个内存数据库,但它提供了持久化机制,可以将内存中的数据保存到磁盘,防止数据丢失。 3. **支持事务...

    阿里巴巴Redis使用规范

    本文将详细介绍阿里巴巴28条Redis使用规范,涵盖了Redis性能优化、数据存储、安全、实例管理等方面的内容。 规范一:控制key的长度 为了避免Redis中的keys过长,阿里巴巴建议控制key的长度,尽量将String类型的数据...

    redis详细使用手册

    9. **性能优化**:Redis的性能优化包括内存管理、网络调优、命令优化等方面,比如设置合适的内存限制、禁用不必要的持久化、使用连接池等。 10. **安全策略**:了解如何配置Redis的访问控制,如设置密码认证、限制...

    Redis 性能优化

    #### 一、Redis内存管理与优化 Redis作为一款高性能的键值存储系统,在处理大量数据时,如何有效地管理和优化内存显得尤为重要。以下几点可以帮助我们更好地理解Redis的内存管理机制以及如何进行优化: 1. **查看...

    Redis新手学习使用介绍

    总的来说,Redis新手入门需要掌握的基本内容包括:Redis的安装与启动、基本数据类型和操作命令、Java客户端的使用、数据持久化与高可用性策略,以及性能优化技巧。通过深入学习和实践,你将能够熟练地在实际项目中...

    REDIS的安装与使用说明

    ### REDIS的安装与使用说明 #### 一、Redis简介 Redis是一种开源的键值存储系统,它提供了多种数据结构的存储方式,并且支持网络、内存数据存储、数据持久化等功能。由于其高性能和灵活性,Redis被广泛应用于缓存...

    Redis基本原理、优化和应用示例.pdf

    - **Redis内存分配原理**:Redis使用jemalloc内存分配器来管理内存。对于String,它直接分配内存;而对于Zipmap,这是一种用于存储小型键值对的内部数据结构,当键值对增多或大小超过一定阈值时,会自动转换为哈希...

Global site tag (gtag.js) - Google Analytics