`
nicegege
  • 浏览: 589377 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

从mysql分页到cassandra,再到redis+cassandra

 
阅读更多

文章转载地址:http://www.cnblogs.com/wuxl360/p/5465670.html

文章写实,很符合技术人员特点。又加上作者的描述清晰易懂。难得的佳作。

分页查询和redis

问题

我在做论坛的是时候遇到了如下的问题。论坛里可以有很多的主题topic,每个topic对应到很多回复reply。现在要查询某个topic下按照replyTime升序排列的第pageNo页的reply,每页pageSize个reply。

reply是存放在mysql中的。以前的实现是利用mysql的limit查询

1
2
3
4
select * from reply
where topicId = ?
order by replyTime asc
limit (pageNo - 1) * pageSize, pageSize

由于现在有很多的主题的回复很多,当有人查询第几百甚至几千页的时候,mysql性能表现很不好。“select limit offset, size” 只要offset太大,传统的关系型数据库的性能表现都不好。

如果能够利用带索引的查询条件先过滤掉一部分数据,就可以大大提高性能,比如:

1
2
3
4
5
select *
from reply
where topicId = ? and replyId > lastReplyIdOfCurrentPage
order by replyTime asc
limit (pageNo - currentPageNo) * pageSize,  pageSize

lastReplyIdOfCurrentPage 是当前页的最后一个reply的id。currentPageNo是当前页的页号。这里用replyId过滤条件,把前面页的内容过滤掉,这样减少了 offset的大小。但是当用户需要跳转到很远的一个页面的时候,offset还是会很大。比如,当前是第10页,要跳转到第1000页,offset = 990 * pageSize,还是会很大,性能依旧不行。尽管目前很多产品,都不提供这样的跳转能力了,但是我们的产品团队还是认为这个功能在我们的产品里面不可或 缺。

迁移到cassandra

后来我们把reply数据全部迁移到了cassandra上。cassandra的数据结构和mysql不一样。我们创建了一个topic_reply 列簇,每一行的行号是topicId,每一列是这个topic的replyId,这样得到类似如下结构

1:1,2,5,33,245,663,780...
2:36,78,89,94,235,345...

在cassandra中列是自然排序的,形成了一个从topic到reply的索引。查询的时候只能查询topicId行的列大于(或小于)replyId的size个replyId,相当于sql:

1
select * from topic_reply where replyId > ? limit size

, 不能够 “limit offset, size”。这意味着如果要查询第一千页,而我不知道第一千页开始的replyId是多少,我就得取出这一千页的数据,这显然是行不通的。所以得想办法从靠近我要取的数据的某个replyId处开始取数据。

redis的SortedSet

无 论是mysql还是cassandra,都不能很好地解决从一个很长的序列中取出任意一段数据的问题,而造成这一问题的根源在于这些数据是存放在磁盘上 的,磁盘不适合做此类的随机读的操作。所以想,如果能有一个程序,管理一些很大很大的放在内存中排序数组就好了,因为对内存中的数组做下标访问,是非常快 速的。做了一下调查正好发现,redis提供了此类的功能。

redis将数据存放到内存中,所以既便是随机读写,速度都是非常快。 redis支持的SortedSet结构正好适合于做分页查询。SortedSet按照给定的score给member排序,允许通过下标或者score 去查询。把同一个topic的replyId作为member,以replyId本身为score存放到SortedSet后,就可以通过下标取值了,例 如:

//存入数据
zadd tr:1 1 1
zadd tr:1 2 2
zadd tr:1 5 5
zadd tr:1 33 33
zadd tr:1 245 245
zadd tr:1 663 663

//pageSize = 3 取 第二页,即下标 3 到5的元素
zrange tr:1 3 5

其中 tr:1 是这个SortedSet的key,”tr:”只是用来区分其它key用的前缀,1是topicId。更详细的内容看redis官网http://redis.io

如此一来,就可以实现任意分页查询了,而且性能非常好。

缓存索引

redis 的数据全部存放到内存中,如果把所有topic到reply的关系都放到内存中,要耗费很多内存,而且这么多的内存实际上很多是浪费的,毕竟大部分的 topic是不活跃的。再者topic到reply的映射关系是非常重要的,所以我们需要把这种关系持久化。最后我们决定,这个映射关系,或者称为索引还 是存放在cassandra里面,只是在需要的时候,才从cassandra里面把索引载入到redis内,然后再利用redis分页查询。如此一 来,redis成了一个支持分页查询的强大的缓存。

分片缓存

对于超长的主题,全新载入到redis一次也是相当的耗时的,我们采取分片来解决这个问题。我们把索引每4800个值分成一片,用另外一个数据结构记录索引长度和索引从第二片开始的每片的开始值。
<iframe id="iframe_0.878788665934626" style="border-width: initial; border-style: none; width: 20px; height: 20px;" src="data:text/html;charset=utf8,%3Cstyle%3Ebody%7Bmargin:0;padding:0%7D%3C/style%3E%3Cimg%20id=%22img%22%20src=%22http://sulong.me/wp-content/uploads/2013/05/index_slice.png?_=5465670%22%20style=%22border:none;max-width:1132px%22%3E%3Cscript%3Ewindow.onload%20=%20function%20()%20%7Bvar%20img%20=%20document.getElementById('img');%20window.parent.postMessage(%7BiframeId:'iframe_0.878788665934626',width:img.width,height:img.height%7D,%20'http://www.cnblogs.com');%7D%3C/script%3E" frameborder="0" scrolling="no"></iframe>

更新的索引的时候更新这个分片信息,记录各分片的头部是为了便于从cassandra载入分片。

查 询的时候把分页查询转化成某个片上某段索引的值。当分片大小大于pageSize并且能被pageSize整除时,这个转化是很简单的,因为分页正好会全 部落在某一个分片中。我们之所以把分片大小设置成4800正是因为这个值能被10 15 20 25 30 40 50 60 80 100 200 等很多常用分页大小整除。分片太大浪费内存,分片太小分片就太多。

只要算出这一页所在的分片,然后把需要的索引段载入到redis,再利用redis的分页查询查出结果。这样,只有活跃的索引分段才会被载入到redis内存中。

如果用mysql来持久化索引效果也是类似的,而且查询更加便利能力更强。

总结

只要产品能接受,就不要使用任意分页,任意跳转。确实需要高速分页查询的时候可以使用redis的SortedSet,但是得注意内存大小和持久化问题。

分享到:
评论

相关推荐

    JavaWeb之Linux与Redis

    通过以上内容的学习,我们可以了解到Java注解的基本用法及其应用场景,Linux系统的基础操作命令,以及Redis和NoSQL数据库的相关概念与使用方法。这些知识对于开发Java Web应用以及管理服务器环境都是非常重要的。

    Google 系统设计真题 20201

    考虑到高并发和实时性,可以使用分布式键值存储系统如BigTable或Cassandra。 3. 时间线生成:用户的时间线是根据他们关注的人的最新推文动态生成的。可以使用队列服务如Kafka或RabbitMQ来处理新推文的发布,并将...

    数据库面试基础知识.docx

    - 通过复制日志(Binlog)实现从主服务器到从服务器的数据同步。 20. **如何优化数据库性能** - **索引**:合理创建索引以加速查询。 - **分库分表**:对大表进行水平拆分,减轻单个表的压力。 - **批量操作**...

    oracle精品教程

    NoSQL数据库则支持各种不同的数据模型,例如键值存储、文档存储、宽列存储和图形数据库,如MongoDB、Redis和Cassandra。 关系理论: 关系理论是数据库领域的重要理论基础,主要包括关系模型、规范化等概念。关系...

    打造自己的RMS框架源代码

    1. 数据存储:RMS框架的基础是数据存储层,通常与关系型数据库(如MySQL、Oracle、SQL Server)或者NoSQL数据库(如MongoDB、Cassandra)配合使用。设计时需要考虑如何抽象数据库操作,创建一个通用的数据访问接口,...

    最全面的sql数据库基础知识笔记

    SQL(Structured Query ...总结,SQL数据库基础知识涵盖了从数据库概念、SQL语法到数据库设计原则、性能优化以及安全措施等多个方面。通过对这些内容的学习,你将能够熟练掌握SQL语言,有效管理和操作数据库系统。

    Spring Data Modern Data Access for Enterprise Java.pdf

    - **定义**:Spring Data是一个基于Spring框架的子项目,旨在简化数据库访问,并支持多种持久化技术,包括但不限于关系型数据库(如MySQL、PostgreSQL等)、NoSQL数据库(如MongoDB、Cassandra等)以及云数据服务。...

    catalogo_jofre

    1. 数据库管理系统:为了存储和组织大量产品数据,通常会使用关系型数据库(如MySQL、Oracle)或NoSQL数据库(如MongoDB、Cassandra)。这些系统提供了高效的数据存储和检索机制。 2. 数据结构与数据模型:设计合适...

    Data:基本资料库

    它可以是关系型数据库(如MySQL、SQL Server)、NoSQL数据库(如MongoDB、Cassandra)或是键值存储(如Redis)。在这里,我们可能指的是使用C#进行开发时的基础数据库操作。 C#是微软公司推出的面向对象的编程语言...

Global site tag (gtag.js) - Google Analytics