的时候,面试官问我,“会MongoDB吗?”,“不会”;“知道redis吗?”,“知道,但是没用过”,“emm……”

在面对高并发的数据读取的时候,当连表查询的需求不是那么的强烈,此时,非关系型数据库得到了高速的发展。其中,非关系型数据库中的键值数据库中的redis便是我们今天需要讲的内容。其中redis为了保证效率,会将数据保存在内存中(当然redis会周期性的把更新的数据写入磁盘或者把修改操作写入追加的记录文件)。

Redis亦被称为数据机构服务器,因为它的值(Value)可以是字符串(String),哈希(Hash),列表(List),集合(Sets)和有序集合(sorted sets)。

Redis 是完全开源免费的,遵守BSD协议,是一个高性能的key-value数据库。

Redis支持数据的持久化,可以将内存中的数据保存在磁盘中,重启的时候可以再次加载进行使用。

Redis不仅仅支持简单的key-value类型的数据,同时还提供list,set,zset,hash等数据结构的存储。

Redis有着更为复杂的数据结构并且提供对他们的原子性操作,这是一个不同于其他数据库的进化路径。Redis的数据类型都是基于基本数据结构的同时对程序员透明,无需进行额外的抽象。

Redis运行在内存中但是可以持久化到磁盘,所以在对不同数据集进行高速读写时需要权衡内存,因为数据量不能大于硬件内存。在内存数据库方面的另一个优点是,相比在磁盘上相同的复杂的数据结构,在内存中操作起来非常简单,这样Redis可以做很多内部复杂性很强的事情。同时,在磁盘格式方面他们是紧凑的以追加的方式产生的,因为他们并不需要进行随机访问。

redis怎么安装,我就不做介绍了,因为不同的系统安装方式不一样。如果不想安装话可以使用在线的redis环境。不过还是推荐一下,因为接下来会使用python去实现一些操作。其中redis数据库的可视化可以使用redis-desktop-manager。命令行操作使用redis-cli。

字符串(Strings)是Redis的基本数据结构之一,由key和value组成。我们可以这样类比成编程语言的变量:keya代表变量名,value代表变量值。

如果我们不希望set的命令覆盖旧值怎么办?在使用NX参数即可。这样,当key存在时,使用set key value NX并不能覆盖原来的值。

在前面我们介绍了字符串类型的Redis储存方案,这个时候我们可以想一想如果我需要储存10w个人的分数,右需要储存10w个人的密码(假如这样做),那么我们需要多少多少个key?20w个key!!那么我们的key又该怎样分配呢(注意:key不能重复)?我们是不是得这样:用户名_score,用户名_pwd,在用户名后面加上不同的类型来代表不同的数据。那么在redis如何解决这些问题呢?使用哈希表!!关于哈希表的数据结构我们可以看看这篇

Redis hash 是一个string类型的field和value的映射表(key任然为key),hash特别适合用于存储对象,每个 hash 可以存储 232 – 1 键值对(43多亿)。使用Hash表不仅能够减少Redis中key的个数,还能优化存储空间,占用的内存要比字符串小很多。

接下来我就不再使用python实现这些东西了,会更多的来介绍各种结构的特点。如果想了解更多的指令可以看菜鸟教程

列表是一种很神奇的结构,可以把列表成一根水管,数据从可以从一边进,然后从另外一边出来(当然,那一边即可以进又可以出,只不过顺序不同而已)。那么这种结构有什么用处呢?我们可以以发消息为例。发消息我们需要保证消息到达的顺序,那么我们是不是就可以使用列表了呢?例如:发送消息从左边进,接受消息从右边得到。下面介绍几个简单的指令:

索引从最左边开始编号,意思就是最后一个lpush的数据的索引是0(将列表想成一个小水管就行)。如果开始索引和结束索引一样,就返回索引位置的值。那么如果从右边开始呢?使用“负索引”即可。其中**-1**代表最右边的数据。-2代表最右边的第二个数据。

这个集合和数学中的集合有着差不多的概念。怎么说呢?在redis的集合中,数据是无序的,不能重复。

前面我们说道,集合是无序的,所以spop的获取也是无序的。(获得数据后会将数据删除)count代表获取几条数据。

这个是获得集合的所有数据(并不会删除数据)。不过这个命令在生产环境中最好不要使用,因为数据量大的话你的服务器可能就炸了。

有序集合,顾名思义就是集合里面的数据是有序的。那么它有什么含义呢?我们想象一下,在一个高并发的场景中,数据是一直更新的,如果我们将数据存到数据库中,如果需要实时获取排名的话,那么肯定会对数据的性能造成很大的影响。毕竟数据量越大,排序时间也就越缓慢。和集合不同的是,有序集合的元素会关联一个double类型的分数,其中元素不能重复,但是分数可以重复。

前面说过,score必须为double类型,所以如果输入非double则会报错。

其中改变量既可以为正数也可以为负数。如果member不存在则会创建,member的分数和改变量一致。

其中使用括号括起来的代表可以省略,如果withsores省略代表只返回值,不返回评分。省略“limit 切片开始位置 结果数量”代表不对结果进行切片。

zrank 和 zrevrank的区别在于,zrank排名是从0开始的,评分越小则排名越靠近0,评分最小的值排名为0。而zrevrank则是评分越大则越靠近0。相同点在于如果值不存在则返回None。

我们来说一个场景,还是以前面的发消息场景为例,如果服务端的消息在进行更新,那么我们如何更新客户端的信息呢?按照前面的方法,我们只有使用轮询查询的方式,按照一定的时间(比如说1s)检查redis,看消息是否发生改变,如果消息发生了改变,则客户端进行更新,那么接下来就会有以下的问题:

其中会返回3条消息:第一条是信息的类型,第二条是频道名,第三条是被发布的内容。其中需要注意,只能接受目前的消息,新加入的订阅是没办法接受到以前的订阅的。

以上便是redis的一个很简单的入门教程。只介绍了redis的简单使用和一些基本操作,至于redis命令中更复杂的指令,百度或者google就行了。对于redis,我觉得我们(作为一个学生)应该关注的是redis为什么能够如此优秀?里面用了什么数据结构,以及如何能够将redis应用到高并发的场景中,怎么实现多服务器数据的保存以及备份问题。

发表回复

您的电子邮箱地址不会被公开。 必填项已用 * 标注