详细讲解redis数据结构(内存模型)以及常用命令
Redis数据类型
与Memcached仅支持简单的key-value结构的数据记录不同,Redis支持的数据类型要丰富得多,常用的数据类型主要有五种:String、List、Hash、Set和Sorted Set。
Redis数据类型内存结构分析
Redis内部使用一个redisObject对象来表示所有的key和value。redisObject主要的信息包括数据类型(type)、编码方式(encoding)、数据指针(ptr)、虚拟内存(vm)等。type代表一个value对象具体是何种数据类型,encoding是不同数据类型在redis内部式。
redisObject 对象示意图
下面分别介绍5种数据类型的用法。
String类型
字符串是Redis值的最基础的类型。Redis中使用的字符串是通过包装的,基于c语言字符数组实现的简单动态字符串(simple dynamic string, SDS)一个抽象数据结构。其源码定义如下:
struct sdshdr { int len; //len表示buf中存储的字符串的长度。 int free; //free表示buf中空闲空间的长度。 char buf[]; //buf用于存储字符串内容。 };
C语言字符串内存结构示意图1
假设上图是”hello”字符串的内存结构,这个时候len=5,free=2那么redis包装后(sds)其长度为:
sizeof(struct sdshdr) + len + free +
其中buf的大小为:
len + free +
1表示1个字节是用来存储结束符’\0’的。Redis字符串是二进制安全的,因为二进制数据通常会有中间某个字节存储’\0’的这种情况,这意味着一个Redis字符串可以包含任何种类的数据,例如一个JPEG图像或者一个序列化的Ruby对象。二进制是否安全,简单的理解就是能不能在字符串中间有‘\0’,如下图:
C语言字符串内存结构示意图2
对于上图,sds认为这个字符串是“hello world”,而C语言的字符处理函数认为这个字符串是“hello”。
应用场景
String是最常用的一种数据类型,普通的key/value存储都可以归为此类。
常用命令
(1)set——设置key对应的值为String类型的value
(2)get——获取key对应的值
192.168.2.129:6379> setnx name lisi (integer) 0 192.168.2.129:6379> setnx name1 wangwu (integer) 1 192.168.2.129:6379> get name "zhangsan" 192.168.2.129:6379> get name1 "wangwu" 192.168.2.129:6379>
(3)mget——批量获取多个key的值,如果可以不存在则返回nil
192.168.2.129:> mget name name1 ) "zhangsan" ) "wangwu" 192.168.2.129:> mget name name1 name2 ) "zhangsan" ) "wangwu" ) (nil) 192.168.2.129:>
(4)incr && incrby——incr对key对应的值进行加加操作,并返回新的值;incrby加指定值
192.168.2.129:> get age "" 192.168.2.129:> incr age (integer) 192.168.2.129:> set age1 "" OK 192.168.2.129:> get age1 "" 192.168.2.129:> incr age1 (integer) 192.168.2.129:> incrby age (integer)
从上面的结果可以看出,我们对int型的age和string型的age1都能进行incr操作时,
实际上type=string代表value存储的是一个普通字符串,那么对应的encoding可以是raw或者是int,如果是int则代表实际redis内部是按数值型类存储和表示这个字符串的,当然前提是这个字符串本身可以用数值表示,比如"20"这样的字符串,当遇到incr、decr等操作时会转成数值型进行计算,此时redisObject的encoding字段为int。如果你试图对name进行incr操作则报错。
192.168.2.129:> incr name (error) ERR value is not an integer or out of range
(5)decr && decrby——decr对key对应的值进行减减操作,并返回新的值;decrby减指定值
192.168.2.129:> decr age (integer) 192.168.2.129:> decrby age (integer) 192.168.2.129:>
(6)其他命令
命令 |
说明 |
setnx |
设置key对应的值为String类型的value,如果key已经存在则返回0 |
setex |
设置key对应的值为String类型的value,并设定有效期 |
setrange |
设置key对应value的子字符串 |
getrange |
获取key对应value的子字符串 |
mset |
批量设置多个key的值,如果成功表示所有值都被设置,否则返回0表示没有任何值被设置 |
msetnx |
同mset,不存在就设置,不会覆盖已有的key |
getset |
设置key的值,并返回key旧的值 |
append |
给指定key的value追加字符串,并返回新字符串的长度 |
strlen |
取指定key的value的长度 |
Hash类型
Hash是一个String类型的field和value之间的映射表,即redis的Hash数据类型的key(hash表名称)对应的value实际的内部存储结构为一个HashMap,因此Hash特别适合存储对象。相对于把一个对象的每个属性存储为String类型,将整个对象存储在Hash类型中会占用更少内存。
Hash 数据类型内部结构示意图
当前HashMap的实现有两种方式:当HashMap的成员比较少时Redis为了节省内存会采用类似一维数组的方式来紧凑存储,而不会采用真正的HashMap结构,这时对应的value的redisObject的encoding为zipmap,当成员数量增大时会自动转成真正的HashMap,此时encoding为ht。
应用场景
用一个对象来存储用户信息,商品信息,订单信息等等。
常用命令
(1)hset——设置key对应的HashMap中的field的value
(2)hget——获取key对应的HashMap中的field的value
192.168.2.129:> hset myhash name zhangsan (integer) 192.168.2.129:> hset myhash age (integer) 192.168.2.129:> hget myhash name "zhangsan" 192.168.2.129:> hget myhash age "" 192.168.2.129:>
(3)hgetall——获取key对应的HashMap中的所有field的value
192.168.2.129:> hgetall myhash ) "name" ) "zhangsan" ) "age" ) "" 192.168.2.129:>
(4)其它命令
命令 说明 hsetnx 设置key对应的HashMap中的field的value,如果不存在则先创建 hmset 批量设置key对应的HashMap中的field的value hmget 批量获取key对应的HashMap中的field的value hincrby 给key对应的HashMap中的field的value加指定的值 hexits 测试key对应的HashMap中的field是否存在 hlen 返回key对应的HashMap中的field的数量 hdel 删除key对应的HashMap中的field hkeys 返回key对应的HashMap中所有的field hvals 返回key对应的HashMap中所有的field的value
List类型
Redis的List类型其实就是每一个元素都是String类型的双向链表。我们可以从链表的头部和尾部添加或者删除元素。这样的List既可以作为栈,也可以作为队列使用。
List数据结构内部示意图
应用场景
如好友列表,粉丝列表,消息队列,最新消息排行等。
常用命令
(1)lpush——在key对应的list的头部添加一个元素。
(2)lrange——获取key对应的list的指定下标范围的元素,-1表示获取所有元素。
(3)lpop——从key对应的list的尾部删除一个元素,并返回该元素。
192.168.2.129:> lpush newlist news1 news2 news3 (integer) 192.168.2.129:> lrange newlist - ) "news3" ) "news2" ) "news1" 192.168.2.129:> lpop newlist "news3" 192.168.2.129:> lrange newlist - ) "news2" ) "news1" 192.168.2.129:>
从上面的操作可以看出,lpush、lpop从表头操作。
(4)rpush——在key对应的list的尾部添加一个元素。
(5)rpop——从key对应的list的尾部删除一个元素,并返回该元素。
192.168.2.129:> rpush newlist2 news1 news2 news3 (integer) 192.168.2.129:> lrange newlist2 - ) "news1" ) "news2" ) "news3" 192.168.2.129:> rpop newlist2 "news3" 192.168.2.129:>
从上面的操作可以看出,rpush、rpop从表尾操作。
(6)其他命令
命令 |
说明 |
linsert |
在key对应的list的特定元素的前或后插入元素 |
lset |
设置key对应的list中指定下标元素的值 |
lrem |
从key对应的list中删除n个和value相同的元素 |
ltrim |
保留key对应的list中指定范围的元素 |
rpoplpush |
从第一个list的尾部移除一个元素并添加到第二个list的头部 |
llen |
返回key对应的list的长度 |
lindex |
返回key对应的list中index的元素 |
Set类型
Redis 集合(Set类型)是一个无序的String类型数据的集合,类似List的一个列表,与List不同的是Set不能有重复的数据。实际上,Set的内部是用HashMap实现的,Set只用了HashMap的key列来存储对象。我们来看看java中HashSet的源码:
public class HashSet<E> extends AbstractSet<E> implements Set<E>, Cloneable, java.io.Serializable { static final long serialVersionUID = -5024744406713321676L; private transient HashMap<E,Object> map; // Dummy value to associate with an Object in the backing Map private static final Object PRESENT = new Object(); /** * Constructs a new, empty set; the backing <tt>HashMap</tt> instance has * default initial capacity (16) and load factor (0.75). */ public HashSet() { map = new HashMap<>(); } ...... /** * Returns an iterator over the elements in this set. The elements * are returned in no particular order. * * @return an Iterator over the elements in this set * @see ConcurrentModificationException */ public Iterator<E> iterator() { return map.keySet().iterator(); }
可见创建一个HashSet的时候实际上创建了一个HashMap;Set中的元素,只是存放在了底层HashMap的key上,底层HashMap的value列为空,遍历HashSet的时候从HashMap中取出keySet来遍历。
Set底层结构示意图
应用场景
集合有取交集、并集、差集等操作,因此可以求共同好友、共同兴趣、分类标签等。
常用命令
(1)sadd——在key对应的set中添加一个元素。
(2)smembers——获取key对应的set的所有元素。
(3)spop——随机返回并删除key对应的set中的一个元素。
192.168.2.129:> sadd myset news1 news2 news3 (integer) 192.168.2.129:> smembers myset ) "news3" ) "news2" ) "news1" 192.168.2.129:> spop myset "news3" 192.168.2.129:>
(4)sdiff——求给定key对应的set与第一个key对应的set的差集
192.168.2.129:> smembers myset ) "news3" ) "news2" ) "news1" 192.168.2.129:> sadd myset2 news3 news4 news5 (integer) 192.168.2.129:> smembers myset2 ) "news4" ) "news3" ) "news5" 192.168.2.129:> sdiff myset myset2 ) "news1" ) "news2" 192.168.2.129:>
(5)suion——求给定key对应的set并集
192.168.2.129:> sunion myset myset2 ) "news3" ) "news1" ) "news2" ) "news4" ) "news5" 192.168.2.129:>
(6)sinter——求给定key对应的set交集
192.168.2.129:> sinter myset myset2 ) "news3" 192.168.2.129:>
(7)其他命令
命令 |
说明 |
srem |
删除key对应的set中的一个元素 |
sdiffstore |
求给定key对应的set与第一个key对应的set的差集,并存储到另一个key对应的set中 |
sinterstore |
求给定key对应的set交集,并存储到另一个key对应的set中 |
suionstore |
求给定key对应的set并集,并存储到另一个key对应的set中 |
somve |
从第一个key对应的set中删除指定元素并添加到第二个key对应的set中 |
scard |
返回key对应的set的元素个数 |
sismember |
测试某个元素是否为key对应的set中的元素个数 |
srandmember |
随机返回key对应的set中的一个元素,但不删除元素 |
SortSet
SortSet顾名思义,是一个排好序的Set,它在Set的基础上增加了一个顺序属性score,这个属性在添加修改元素时可以指定,每次指定后,SortSet会自动重新按新的值排序。
sorted set的内部使用HashMap和跳跃表(SkipList)来保证数据的存储和有序,HashMap里放的是成员到score的映射,而跳跃表里存放的是所有的成员,排序依据是HashMap里存的score。
应用场景
如按时间排序的时间轴。
常用命令
(1)zadd ——在key对应的zset中添加一个元素
(2)zrange——获取key对应的zset中指定范围的元素,-1表示获取所有元素
192.168.2.129:> zadd myzset "one" "two" "three" (integer) 192.168.2.129:> zrange myzset - ) "one" ) "two" ) "three" 192.168.2.129:> zrange myzset - withscores ) "one" ) "" ) "two" ) "" ) "three" ) "" 192.168.2.129:>
(3)zrem——删除key对应的zset中的一个元素
192.168.2.129:> zrem myzset one (integer) 192.168.2.129:> zrange myzset - withscores ) "two" ) "" ) "three" ) "" 192.168.2.129:>
(4)其它命令
命令 说明 zincrby 如果key对应的zset中已经存在元素member,则对member的score属性加指定的值 zrank 返回key对应的zset中指定member的排名。其中member按score值递增(从小到大);排名以0为底,也就是说,score值最小的成员排名为0 zrevrank 获得成员按score值递减(从大到小)排列的排名 zrevrange 返回有序集key中,指定区间内的成员。其中成员的位置按score值递减(从大到小)来排列 zrangebyscore 返回有序集key中,指定分数范围的元素列表 zcount 返回有序集key中,score值在min和max之间(默认包括score值等于min或max)的成员 zcard 返回key的有序集元素个数
键值常用命令
keys/exits/del/expire/ttl/move/persist/randomkey/rename/type
服务器常用命令
ping/echo/select/quit/dbsize/info/config get/flushdb/flushall
这些命令都很容易使用,就不举例说明了。到此,redis的数据类型以及常用命令已经介绍完毕,下一篇我们将学习redis的一些高级特性。
详细讲解redis数据结构(内存模型)以及常用命令的更多相关文章
- 分布式缓存技术redis学习系列(二)——详细讲解redis数据结构(内存模型)以及常用命令
Redis数据类型 与Memcached仅支持简单的key-value结构的数据记录不同,Redis支持的数据类型要丰富得多,常用的数据类型主要有五种:String.List.Hash.Set和Sor ...
- 分布式缓存技术redis学习(二)——详细讲解redis数据结构(内存模型)以及常用命令
Redis数据类型 与Memcached仅支持简单的key-value结构的数据记录不同,Redis支持的数据类型要丰富得多,常用的数据类型主要有五种:String.List.Hash.Set和Sor ...
- RedisTemplate访问Redis数据结构(介绍和常用命令)
Redis 数据结构简介 Redis 可以存储键与5种不同数据结构类型之间的映射,这5种数据结构类型分别为String(字符串).List(列表).Set(集合).Hash(散列)和 Zset(有序集 ...
- 分布式缓存技术redis系列(二)——详细讲解redis数据结构(内存模型)以及常用命令
https://www.cnblogs.com/hjwublog/p/5639990.html
- 深入了解一下Redis的内存模型!
一.前言 Redis是目前最火爆的内存数据库之一,通过在内存中读写数据,大大提高了读写速度,可以说Redis是实现网站高并发不可或缺的一部分. 我们使用Redis时,会接触Redis的5种对象类型(字 ...
- Redis启动服务和String常用命令
Redis启动服务和String常用命令 1. 启动Redis服务 E:\redis>redis-server.exe redis.windows.conf _._ _.-``__ ''-._ ...
- redis系列之2----详细讲解redis数据结构(内存模型)以及常用命令
Redis数据类型 与Memcached仅支持简单的key-value结构的数据记录不同,Redis支持的数据类型要丰富得多,常用的数据类型主要有五种:String.List.Hash.Set和Sor ...
- 详细讲解 Redis 的两种安装部署方式
Redis 是一款比较常用的 NoSQL 数据库,我们通常使用 Redis 来做缓存,这是一篇关于 Redis 安装的文章,所以不会涉及到 Redis 的高级特性和使用场景,Redis 能够兼容绝大部 ...
- redis五种数据类型和常用命令及适用场景
一.redis的5种数据类型: 1.基础理解: string 字符串(可以为整形.浮点型和字符串,统称为元素) list 列表(实现队列,元素不唯一,先入先出原则) set 集合(各不相同的元素) h ...
随机推荐
- [转帖]ASML EUV光刻机累计生产450万块晶圆:一台12亿元
ASML EUV光刻机累计生产450万块晶圆:一台12亿元 来源驱动之家 ...网页被我关了 就这样吧. 截至目前,华为麒麟990 5G是唯一应用了EUV极紫外光刻的商用芯片,台积电7nm EUV工艺 ...
- mac下安装和运行redis
第一部分:安装 第一步:直接从官网下载**.tar.gz这个包.并用tar -zxvf **.tar.gz -C 指定目录 示例命令: tar -zxvf redis-5.0.5.tar.gz ...
- centos安装sftp服务
一.创建sftp服务数据目录及相关测试用户 [root@localhost ~]# mkdir -pv /data/sftp/ #sftp数据目录 [root@localhost ~]# chown ...
- 用vscode开发vue应用
阅读 3237 收藏 205 2019-05-02 原文链接:segmentfault.com 云服务器 1 核 2G , 9元/月 ,买十送二,99/年!!!快来上车!developer.huawe ...
- JavaScriptCore在浏览器引擎中的位置
因为随着JS这门语言的发展,JS的宿主越来越多,有各种各样的浏览器,甚至是常见于服务端的Node.js(基于V8运行). 2. Webkit 源代码由三大模块组成: 1). WebCore ...
- Linux 笔记 - 第二十二章 Nginx 配置 SSL
一.前言 基础知识 1.1 公钥密码体制(public-key cryptography) 公钥密码体制分为三个部分,公钥.私钥.加密解密算法,它的加密解密过程如下: 加密:通过加密算法和公钥对内容( ...
- 2 java并行基础
我们认真研究如何才能构建一个正确.健壮并且高效的并行系统. 进程与线程 进程(Process):是计算机中的程序关于某数据集合上的一次运行活动,是系统进行资源分配和调度的基本单位,是操作系统结构的基础 ...
- 阿里云监控RDS
RDS性能监控API https://help.aliyun.com/document_detail/26280.html?spm=a2c4g.11186623.6.1576.341d7159uzLD ...
- C++运算符重载学习总结
在C ++中,我们可以使运算符适用于用户定义的类. 这意味着C ++能够为运算符提供数据类型的特殊含义,这种能力称为运算符重载. 例如,我们可以在像String这样的类中重载运算符'+',这样我们就可 ...
- Windows 获取进程ID
DWORD GetProcessID(const char *ProcessName) { PROCESSENTRY32 pe32; pe32.dwSize = sizeof(PROCESSENTRY ...