作为一个反主流的开发者,在某种程度上,我对传统关系型数据库一直有点“讨厌”,因为关系型数据库实际上和面向对象思想是完全冲突的,前者建立在数学集合理论的基础上,而后者则是建立在软件工程基本原则的基础上。虽然传统的ORM、序列化/反序列化在一定程度上解决了这种冲突,但是软件开发中关于使用原生SQL语句还是使用ORM框架的争论从来没有停止过。可是实际的业务背景中,是完全无法脱离数据库的,除非在某些特定的场合下,考虑到信息安全因素而禁止开发者使用数据库,在主流技术中数据库是一个非常重要的组成部分。为了弥补这个技术上的短板,从这篇文章开始,我将会学习一个经典的缓存技术:Redis。我们这里将Redis定性为一门缓存技术,这说明Redis和MySQL等主流的数据库存在本质上的区别,那么这些区别到底在哪里呢?或许在看完这个系列文章以后,你心中自然就会有了答案。

Redis是什么?

  Redis是什么?这是本文第一个问题。Redis是一个开源的使用ANSI C语言编写的、支持网络、 基于内存的、支持持久化的日志型、Key-Value数据库。从如此丰富的修饰语中,我们基本可以抽离出这些信息:

  • Redis是一个Key-Value存储系统
  • Redis的数据全部缓存在内存里
  • Redis可以通过网络实现主从同步
  • Redis支持丰富的数据类型可实现持久化

  那么该如何给Redis一个准确的定义呢?或许这个定义可以帮助我们更好的理解Redis,即Redis是一个高性能的Key-Value数据库。我们知道主流的数据存储方案,可以分为关系型数据库和非关系型数据库两大类。传统的Oracle、MySQL和SQLServer都是关系型数据库,关系型数据库将复杂的数据结构归结为二元关系,即二维表形式,而对数据的操作则建立在一个或多个关系表格中,并通过这些表格间的分类、合并、连接和选取等运算实现数据处理。如同天地万物,有阴影的地方就会有阳光。和关系型数据库相对应的,我们称之为非关系型数据库,这是一个泛指的概念。实际上非关系型数据库,根据设计原理的不同,具体可分为:键-值存储数据库、列存储数据库、文档数据库和图数据库四种。我们通常称非关系型数据库为NoSQL,即”Not Only SQL”,从这个概念我们或许可以明白,SQL和NoSQL并非是完全对立的两个世界,它们各自在其擅长的应用场景中发挥着重要的作用。

  所以我选择Redis这样一个非关系型数据库,从某种意义上来说,我是想说明一件事情,数据库技术并非绝对代表着关系型数据库和SQL,实际上SQL这门语言存在一定缺陷,就像我们提及Web技术常常想到是如何去做一个网站(MVC),可你同样会意识到Web API是更为重要的Web技术。这个世界并非是一成不变的,每一天都是新的挑战。

开始使用Redis

  好了,在了解了Redis是一个什么东西以后,现在我们来正式开始使用Redis。Redis作为一个开源的键-值数据库,我们可以从它的官方网站或者是从Github来获取。这里推荐从官方网站下载相对稳定的版本,这里博主选择的是3.2.8版本,需要注意我们这里从官方网站下载的是源代码版本,所以首先第一件事情就是编译源代码。如果你非常擅长在Window下编译类似项目,可以尝试在Windows下进行编译。博主这里推荐大家使用Linux或者MacOS来编译,因为主流开源项目使用的Makefile都是Unix世界里的产物,所以使用Linux或者MacOS能够为我们节省大量的时间。博主这里使用的是Elementary OS这个Linux发行版(对应Ubuntu14),编译方法如下:

Redis的编译与安装

1
2
3
4
$ wget http://download.redis.io/releases/redis-3.2.8.tar.gz
$ tar xzf redis-3.2.8.tar.gz
$ cd redis-3.2.8
$ make

  在这里,除了make的步骤严格依赖命令行以外,其余的步骤都可以手动完成,所以因为惧怕命令行而不愿意接触Linux的世界,事实证明,对一个真正的程序员来讲,命令行是一个唯一可以让人不被外界所干扰的高效地工具,Git是这个世界上最好没有之一的版本控制工具,如果你喜欢Git,那么你更应该尝试去喜欢Linux。好了,在完成对Redis的编译后,我们就可以开始使用Redis了。Redis是一个C/S架构的键-值数据库,这意味着我们需要Redis的服务端程序和客户端程序。在完成编译以后,我们将得到redis-server和redis-cli这两个内置服务端程序和客户端程序。实际使用中我们会接触到不同语言下的redis客户端,在这里我们直接使用Redis内置的客户端:

1
2
3
4
//开启Redis服务
$ src/redis-server
//开启Redis客户端
$ src/redis-cli

  需要注意的是在这里服务端和客户端,是在两个不同的终端窗口中运行的,当我们看到下面的窗口时,即表明Redis服务开启就绪,此时我们就可以通过客户端来输入各种命令来完成数据的存取,默认情况下Redis每次会随机分配一个端口,这里Redis采用6379端口进行通信:

Redis服务
Redis服务

  Redis是一个采用键-值存储方案的数据库,因而传统关系型数据库里的SQL在这里将不再适用。你可以将Redis理解为一个字典,我们可以向这个字典中储存任何Redis支持的数据类型,并通过键名来获取字典中存储的对应数值。我们来看下面的例子,以下命令均在redis-cli中执行:

1
2
3
4
SET foo bar
OK
GET foo
"bar"

Redis中支持的数据类型

  这个例子演示了如何在Redis中存储和读取一个简单的字符串类型的值,看起来这一切都非常简单啊,的确Redis就是这样一个简单而高效的键-值数据库。我们在前面提到Redis支持各种各样的数据类型,那么它到底支持哪些数据类型呢?具体来讲,Redis支持5种基本的数据类型:

  • 字符串(Strings):最基本的数据类型,使用SET/GET命令来存储和读取字符串类型的值。在Redis中最多可支持512兆字节的字符串长度,这意味着我们可以常见的数据类型序列化后再存储到Redis中。
  • 散列/哈希(Hashes):专门用来表示对象的数据类型。散列/哈希是键-值对的集合,可以维系字符串字段和字符串值间的映射关系,因此它主要用来表示对象。在Redis中可以使用HMSET、HMGET、HGET、HGETALL四种命令来存储和读取散列类型的值。
  • 列表(Lists):指按照插入顺序排序的字符串元素的集合,特别地,Redis中的列表是采用链表实现的,因为对数据库系统而言,一个非常重要的特性是可以支持在含有大量元素的集合中快速添加元素。常见的应用于列表的命令主要有LPUSH、RPUSH、LPOP、RPOP和LRANGE。
  • 集合(Sets):指不重复且无序的字符串元素的集合。针对列表,常见的命令主要有:SADD、SPOP、SCARD、SMEMBERS和SISMEMBER。例如SADD命令可以向集合中添加元素,SPOP命令可以从集合中删除元素,SCARD命令可以返回集合内元素个数,SMERMERS命令可以枚举集合的所有元素,SISMEMBER命令可以判断指定元素是否在指定集合内。
  • 有序集合(Sorted Sets):有序集合与集合相似,不同点在于集合中的每一个元素都会关联一个浮点型的数值,该数值称为score,事实上Redis正是根据score来对集合内的元素进行排序的。集合内的元素是不允许重复的,但是score是可以允许重复的。常见的命令有:ZADD、ZCARD、ZCOUNT、ZREM、ZSCORE等。

Redis中和键有关的命令

  我们知道Redis是一个键-值数据库,所以在了解了Redis中支持的数据类型,即“值”以后,现在让我们将关注点回归到“键”上面来,这是因为作为一个键-值数据库,键是我们从数据库中获取值的唯一方式,因此在这里说说Redis中那些和键有关的命令,这些命令基本都遵循下面的命名格式,常见的命令有:

1
COMMAND KEY_NAME
  • DEL:该命令将在键名存在时从数据库中删除指定键,成功则返回1,否则返回0。
  • DUMP:该命令将序列化指定键,并返回被序列化的值。
  • EXISTS:该命令用以判断指定键是否存在。
  • EXPIRE:该命令用以给指定键设置过期时间。
  • KEYS:该命令用以返回所有满足匹配模式的键。
  • PERSIST:该命令用以移除指定键的过期时间。
  • RENAME:该命令用以重命名指定键。

​ 好了,这就是这篇博客的内容了,自我感觉Redis中的内容相对分散,这种细小的知识点都隐藏在命令中,最初在介绍不同的数据类型的时候,在文章中均做了详细的介绍并辅以终端脚本,可是最后发现这样写下去还不如去看官方文档,像Redis这种即使学习了都不见得有机会使用的技术,当然我并不是说Redis不好啊,关系型数据库目前依然是主流的技术驱动力量,所以我觉得我们学习的时候最好是“观其大略”、“不求甚解”,首先注重整体知识体系上的理解,微枝末叶上的细节问题可以在使用的时候去查阅文档。在下面的文章中,我重点关注的内容是Redis的事务脚本发布/订阅不同语言下Redis的使用,希望大家继续关注我的博客,本篇结束!