redis干啥的,一般人都知道,但很多人只知道是个缓存数据库,其它的就不知道了,本猿无能亦是如此,然知耻而后勇,我们该理一理这里边的一些逻辑,看看redis究竟是怎么一回事儿,能干啥,怎么做的,这样才能心中有数,用到的时候或者进行技术选型的时候胸有成竹心里不慌,这是一个技术人员应有的自我修养。就从最基本的数据类型开始吧!
有5种数据类型:string、hash、list、set跟sortedset
但这些只是redis对外的数据结构,实际上每种数据结构都有自己底层的内部编码实现,而且是多种实现,这样redis会在何时的场景选择合适的内部编码。
----------------------------------------------------------------------------------------------------
1、string
java中string底层是一个char[],那redis中是不是也是一个char[]呢?
上图是http://doc.redisfans.com/中的命令列表,如果string的底层是一个char[]的话,那么append操作就要进行扩容,strlen操作就要遍历或者把长度存起来(类似于ArrayList的size),那么整体来说,string应该就是一个char[]的封装。实际的确是类似的。源码解析:http://www.cnblogs.com/huangxincheng/p/4968085.html
内部编码:
int: 8个字节的长整型
embstr: 小于等于39个字节的字符串
raw: 大于39个字节的字符串
redis会根据当前值的类型和长度决定使用哪种内部编码实现。
可以看到,在value为纯数字的情况下int可存放的范围是非常广的,千万亿级别的都没问题;而如果是非数字字符,那么直接就是embstr,如果字符长度大于39,那就是raw了。这个raw是个动态字符串,其实就是我们上边说的char[]的封装。
string类型适合用来存储数字或者字符串;对象序列化为字符串后也可以用来存储对象。
注意:字符串最大可存储512M,由于key也是字符串,所以key最大值也是512M。redis的其它数据类型均没有限制对象大小,也就是理论上只要小于物理内存即可;
跟java类比,java中integer最大值2^32=2147483647,这也是数组的最大长度,超过长度编译会报错;,2^32个byte是4G,一个char占2字节,那么java中char[]若按内编码计算,则最大占内存为8G;C语言中char占1个字节,最大512M,那么char[]长度应为integer最大值的1/8也就是2^29(倒着推的,没看源码,可能有误)。
常用操作:
存取跟批量存取:get,set,mget,mset,setnx,setex,getbit,setbit
对字符串的追加,截取:append,getrange,setrange
对数字的加减:incr,incrby,decr,decrby,incrbyfloat
2、hash
hash适合用来存储对象:
内部编码有2种:
ziplist:压缩列表,有点类似于数组(但不是),内存连续,而且压缩了前驱后驱指针;当hash类型元素个数小于hash-max-zip-entries配置(默认512),同时所有值都小于hash-max-ziplist-value(默认64)时,使用ziplist作为哈希的内部实现。
hashtable:哈希表,不多解释,数组加链表结构。
假设值都满足条件,那么元素个数在512之前都是ziplist实现,查找效率其实是不高的,不过由于内存连续,可以根据结构找到下个元素,遍历起来也还凑合;如果一定要给一个使用ziplist的理由,应该就是redis实在是想尽量省一点内存吧;
两个条件,一旦有一个不达标,就会转为hashtable,数组加链表增删很容易,但略费内存;
当一个hash的编码由ziplist变成hashtable后,即使再替换掉所有值,它也一直会是hashtable类型,变不回来了。
同时存取100w条数据,两者耗时接近,string存取快,时间在110-120s,但占用内存比hash多20%左右;hash占用内存少,但耗时在120-140s左右,耗时比string多20%;究其原因,应该是string除了记录数据还要记录对象类型信息,因此消耗空间多,而hash因为要组装对象数据,耗时略长;
常用操作:
存取:hset,hget,hmset,hmget,hgetall,hsetnx
判断是否存在:hexists
3、list
redis的list是简单的字符串列表,按照插入顺序排列;可存放(2^32-1)个元素。
内部编码2种:
ziplist: 跟hash类似,也是两个参数配置,只是名字hash改为了list;
linkedlist:当不满足配置时会转换成linkedlist,顾名思义双向链表,转换会同样跟hash一样,不可逆。
list中保存了head,tail节点以及长度值len,也就是操作头尾以及查看list元素个数,时间复杂度为O(1),恒定的;
ps:redis3.1以及之前的版本,list的操作是以上所述;从redis3.2开始,做了优化,引入了quicklist,以上两个配置作废,默认使用quicklist;官方的对quicklist的解释是a doubly linked list of ziplists,也就是一个由ziplist组成的双向链表。
list常用的方法:
压入弹出:LPOP,LPUSH,RPOP,RPUSH。从名字看出四个分别是从左边弹出,压入,从右边弹出,压入。也就是两端开口的一个队列。
长度,取值等:llen,lindex,ltrim(删除不在区间的元素)
4、set
redis的set是string类型的无序集合,成员不能重复,底层实现主要是哈希表,最大长度(2^32-1)。
长度限制跟list相同,除了支持增删改查还支持集合交集,并集,补集;
内部编码2种:
intset:当集合中元素都是整数而且元素个上诉小于set-max-intset-entries(默认512)时,redis会选用intset作为集合内部实现;
hashtable:不满足上述情况则采用hashtable,跟hash的情况相比,就是val是null而已,只存了key。
这里要注意的是此处的set跟java中的set略有区别,这里set中的元素都是字符串,而java存放的是对象,这点不太一样;
常用操作,sadd,spop,srem
sadd:添加,sadd fruits pear apple banana tomato grape orange
spop:移除并返回集合中一个随机元素
srem:移除集合中一个或多个元素
5、zset
zset存放的是一个数值跟一个字符串,成员不重复,最大长度跟set一样;
其实我们已经发现,redis的数据类型跟我们直观上认识的java并不一致!
内部编码2种:
ziplist:一样的配置,只是名字改成了zset,默认值128,对value的闲置改成了默认64字节;
skiplist:跳跃表,ziplist条件不满足的时候,使用跳跃表作为内部实现;
默认按key升序排列:
可以看到排列顺序为升序,而且小数有变化
常用操作:
添加删除:zadd,zrem
获取区间数据:zrange,获取排名:zrank,返回值:zscore,值递增:zincrby
集合间操作,子交并补:zunionstore,zinterstore
zset的使用场景:由于支持排序,以及集合的相关操作,所以在关注列表,共同关注人等操作上比较适用。
-------------------------------------
学习redis随手笔记
- Redis数据类型和常用命令
Redis相较于其它的数据库虽然简单,但是要熟记所有命令的用法也并非易事.一个简单的技巧是通过要操作的数据类型来将这些命令进行结构化. 数据类型和对应命令 所有存储于redis中的数据都对应于一个键值 ...
- redis数据类型及常用命令介绍(图文实例)
aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAhgAAAFLCAYAAACUdvXUAAAgAElEQVR4nO3da1da58L2fT5KfPvc96 ...
- redis笔记总结之redis数据类型及常用命令
三.常用命令 3.1 字符串类型(string) 字符串类型是Redis中最基本的数据类型,一个字符串类型的键允许存储的数据的最大容量为512MB. 3.1.1 赋值与取值: SET key valu ...
- redis 介绍和常用命令
redis 介绍和常用命令 redis简介 Redis 是一款开源的,基于 BSD 许可的,高级键值 (key-value) 缓存 (cache) 和存储 (store) 系统.由于 Redis 的键 ...
- 记录redis安装及常用命令
Redis安装及常用命令 一.安装 1.下载,解压,进入redis解压目录,make. make PREFIX=目录/redis install :安装到指定目录文件名为redis. 2.将解压目录里 ...
- redis五种数据类型和常用命令及适用场景
一.redis的5种数据类型: 1.基础理解: string 字符串(可以为整形.浮点型和字符串,统称为元素) list 列表(实现队列,元素不唯一,先入先出原则) set 集合(各不相同的元素) h ...
- Redis的数据类型及其常用命令
快速入门Redis 首先安装redis: windows下安装redis Linux下安装redis 1. 什么是redis Redis属于nosql(非关系型数据库) 关系型数据库是基于关系表的数据 ...
- VMWare12安装CentOS7以及redis安装和常用命令
一.vmware安装centos7后没有网卡 VMWare 12版本不提供32位安装程序,只有64位安装程序,如果在安装CentOS时,选择的是CentOS而不是CentOS 64位,则会出现Cent ...
- 数据库之redis篇(2)—— redis配置文件,常用命令,性能测试工具
redis配置 如果你是找网上的其他教程来完成以上操作的话,相信你见过有的启动命令是这样的: 启动命令带了这个参数:redis.windows.conf,由于我测试环境是windows平台,所以是这个 ...
随机推荐
- 【转】C#对XML文件的各种操作实现方法
[转]C#对XML文件的各种操作实现方法 原文:http://www.jb51.net/article/35568.htm XML:Extensible Markup Language(可扩展标记语言 ...
- python--输出spwm的数组
python的功能是非常强大的,这个例子使用python编写的输出spwm数组(不对,在C语言或者其他语言叫做数组,在这里叫做list.)的程序,我们在单片机程序里调用这个程序,可以达到输出spwm波 ...
- Android camera调用出现错误解决方法
开发时,先是使用三星的手机测试,发现一切正常: 但是到了小米的手机的时候,发现图片很模糊,发现是设置camera.setParameters(parameters);报错导致用的是默认的最小的分辨率, ...
- 关于eclipse导入maven项目
1:删除其他的配置文件,只需要源码 和 pom文件 2:导入项目,再修改几个地方: 2.1: 所选项目右键- properties - Project Facet,勾上 Dynamic Web Mod ...
- sql 插入
今天处理了一个有关数据库表数据批量插入的问题.部分细节,自己之前没有遇到过.索性就整理下来,做个备忘录. 主要是将一个表的数据导入到另一张表中.这种插入方法,需注意两张表的对于字段的数据结构需要保持一 ...
- linux linux系统的安装及使用
linux linux系统的安装及使用 一.linux系统中安装vm-tools工具: 步骤: 1.在vmware workstation软件中:虚拟机-安装vmware-tools-状态栏会提示- ...
- 求解热传导方程matlab
这是非稳态一维热传导的方法,也叫古典显格式. 如果是做数学建模,就别用了,这种方法计算量比较大,算的很慢,而且收敛不好. 但是如果实在没办法也能凑合用. 该改的地方我都用???代替了. 给个详细解释h ...
- bootstrap添加模态窗后,再弹出消息提示框后,原先的滚动条消失
设置需要滚动的模态框 overflow :scroll
- 字串变换 bfs + 字符串
题目描述 已知有两个字串A,BA,BA,B及一组字串变换的规则(至多666个规则): A1A_1A1 ->B1 B_1B1 A2A_2A2 -> B2B_2B2 规则的含义为:在 ...
- Til the Cows Come Home (dijkstra算法)
Bessie is out in the field and wants to get back to the barn to get as much sleep as possible before ...