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 + 1

其中buf的大小为:

len + free + 1

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) 0192.168.2.129:6379> setnx name1 wangwu(integer) 1192.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:6379> mget name name11) "zhangsan"2) "wangwu"192.168.2.129:6379> mget name name1 name21) "zhangsan"2) "wangwu"3) (nil)192.168.2.129:6379>

4.incr && incrby:incr对key对应的值进行加加操作,并返回新的值;incrby加指定值

192.168.2.129:6379> get age"20"192.168.2.129:6379> incr age(integer) 21192.168.2.129:6379> set age1 "20"OK192.168.2.129:6379> get age1"20"192.168.2.129:6379> incr age1(integer) 21192.168.2.129:6379> incrby age 3(integer) 24

从上面的结果可以看出,我们对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:6379> incr name(error) ERR value is not an integer or out of range

5.decr && decrby:decr对key对应的值进行减减操作,并返回新的值;decrby减指定值

192.168.2.129:6379> decr age(integer) 23192.168.2.129:6379> decrby age 3(integer) 20192.168.2.129:6379>

6.其他命令

Hash类型

Hash是一个String类型的field和value之间的映射表,即redis的Hash数据类型的key(hash表名称)对应的value实际的内部存储结构为一个HashMap,因此Hash特别适合存储对象。相对于把一个对象的每个属性存储为String类型,将整个对象存储在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:6379> hset myhash name zhangsan(integer) 1192.168.2.129:6379> hset myhash age 20(integer) 1192.168.2.129:6379> hget myhash name"zhangsan"192.168.2.129:6379> hget myhash age"20"192.168.2.129:6379>

3.hgetall:获取key对应的HashMap中的所有field的value

192.168.2.129:6379> hgetall myhash1) "name"2) "zhangsan"3) "age"4) "20"192.168.2.129:6379>

4.其它命令

List类型

Redis的List类型其实就是每一个元素都是String类型的双向链表。我们可以从链表的头部和尾部添加或者删除元素。这样的List既可以作为栈,也可以作为队列使用。

应用场景

如好友列表,粉丝列表,消息队列,最新消息排行等。

常用命令

1.lpush:在key对应的list的头部添加一个元素。

2.lrange:获取key对应的list的指定下标范围的元素,-1表示获取所有元素。

3.lpop:从key对应的list的尾部删除一个元素,并返回该元素。

192.168.2.129:6379> lpush newlist news1 news2 news3(integer) 3192.168.2.129:6379> lrange newlist 0 -11) "news3"2) "news2"3) "news1"192.168.2.129:6379> lpop newlist"news3"192.168.2.129:6379> lrange newlist 0 -11) "news2"2) "news1"192.168.2.129:6379>

从上面的操作可以看出,lpush、lpop从表头操作。

4.rpush:在key对应的list的尾部添加一个元素。

5.rpop:从key对应的list的尾部删除一个元素,并返回该元素。

192.168.2.129:6379> rpush newlist2 news1 news2 news3(integer) 3192.168.2.129:6379> lrange newlist2 0 -11) "news1"2) "news2"3) "news3"192.168.2.129:6379> rpop newlist2"news3"192.168.2.129:6379>

从上面的操作可以看出,rpush、rpop从表尾操作。

6.其他命令

Set类型

Redis 集合(Set类型)是一个无序的String类型数据的集合,类似List的一个列表,与List不同的是Set不能有重复的数据。实际上,Set的内部是用HashMap实现的,Set只用了HashMap的key列来存储对象。我们来看看java中HashSet的源码:

public class HashSet    extends AbstractSet    implements Set, Cloneable, java.io.Serializable{    static final long serialVersionUID = -5024744406713321676L;    private transient HashMap 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 HashMap 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 iterator() {        return map.keySet().iterator();    }

可见创建一个HashSet的时候实际上创建了一个HashMap;Set中的元素,只是存放在了底层HashMap的key上,底层HashMap的value列为空,遍历HashSet的时候从HashMap中取出keySet来遍历。

应用场景

集合有取交集、并集、差集等操作,因此可以求共同好友、共同兴趣、分类标签等。

常用命令

1.sadd:在key对应的set中添加一个元素。

2.smembers:获取key对应的set的所有元素。

3.spop:随机返回并删除key对应的set中的一个元素。

192.168.2.129:6379> sadd myset news1 news2 news3(integer) 3192.168.2.129:6379> smembers myset1) "news3"2) "news2"3) "news1"192.168.2.129:6379> spop myset"news3"192.168.2.129:6379>

4.sdiff:求给定key对应的set与第一个key对应的set的差集

192.168.2.129:6379> smembers myset1) "news3"2) "news2"3) "news1"192.168.2.129:6379> sadd myset2 news3 news4 news5(integer) 3192.168.2.129:6379> smembers myset21) "news4"2) "news3"3) "news5"192.168.2.129:6379> sdiff myset myset21) "news1"2) "news2"192.168.2.129:6379>

5.suion:求给定key对应的set并集

192.168.2.129:6379> sunion myset myset21) "news3"2) "news1"3) "news2"4) "news4"5) "news5"192.168.2.129:6379>

6.sinter:求给定key对应的set交集

192.168.2.129:6379> sinter myset myset21) "news3"192.168.2.129:6379>

7.其他命令

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:6379> zadd myzset 1 "one" 2 "two" 3 "three"(integer) 3192.168.2.129:6379> zrange myzset 0 -11) "one"2) "two"3) "three"192.168.2.129:6379> zrange myzset 0 -1 withscores1) "one"2) "1"3) "two"4) "2"5) "three"6) "3"192.168.2.129:6379>

3.zrem:删除key对应的zset中的一个元素

192.168.2.129:6379> zrem myzset one(integer) 1192.168.2.129:6379> zrange myzset 0 -1 withscores1) "two"2) "2"3) "three"4) "3"192.168.2.129:6379>

4.其它命令

Redis常用命令

键值常用命令

keys/exits/del/expire/ttl/move/persist/randomkey/rename/type

服务器常用命令

ping/echo/select/quit/dbsize/info/config get/flushdb/flushall

这些命令都很容易使用,就不举例说明了。到此,redis的数据类型以及常用命令已经介绍完毕,下一篇我们将学习redis的一些高级特性。

Java知音,专注于Java实用文章推送,不容错过!

来源:cnblogs.com/hjwublog/p/5639990.html

redis list放入对象_Redis从入门到入土:详细讲解内存模型以及常用命令相关推荐

  1. 【Ubuntu入门到精通系列讲解】远程管理常用命令速查

    目标 关机/重启 shutdown 查看或配置网卡信息 ifconfig ping 远程登录和复制文件 ssh scp 01. 关机/重启 序号 命令 对应英文 作用 01 shutdown 选项 时 ...

  2. Redis学习笔记①基础篇_Redis快速入门

    若文章内容或图片失效,请留言反馈.部分素材来自网络,若不小心影响到您的利益,请联系博主删除. 资料链接:https://pan.baidu.com/s/1189u6u4icQYHg_9_7ovWmA( ...

  3. java怎么把数据封进对象里_(Java)想把数组中一条一条的数据全部放入对象中去..要怎么做呢...

    展开全部 Java把数组中一条一条的数据全部放入对象中去操作如下: 1.先获取到数组中的数据dataArray数据. 2.接32313133353236313431303231363533e58685 ...

  4. HTML5div css入门案例,div+css教学教案(学习入门到精通~)详细讲解.doc

    | div+cssHYPERLINK "/"教程(入门到精通) 目录: 一.div+css教程(入门到精通)详细讲解 二.DIV+CSS网页布局常用基础知识 三.div+css常用 ...

  5. redis放入对象的几种方式

    redis中存放对象的方式: 1.hash:可放入简单对象,复杂对象没试过 2.json:可放入复杂对象,复杂对象也可以放入 3.序列化:可放入简单对象,复杂对象放不进去,自己试过(通过springb ...

  6. redis储存实体类对象_Redis如何存储对象与集合示例详解

    前言 大家都知道在项目中,缓存以及mq消息队列可以说是不可或缺的2个重要技术.前者主要是为了减轻数据库压力,大幅度提升性能.后者主要是为了提高用户的体验度,我理解的是再后端做的一个ajax请求(异步) ...

  7. csdn入门测试教程------mysql数据库命令大全以及常用命令 安装教程 基础知识 附【练习题】

    前言: mysql数据库是每一个测试小白入行必学的一个知识,学会这些命令呢也能让你在玩转数据库的时候给你提供很多的便利也是很不错的,最后希望各位小伙伴学有所成,心想事成咯, 废话不多说直接上正文. [ ...

  8. Java基础18-String类【String类的特点对象个数常用方法】【超详细讲解】

    Java基础-String类[超详细讲解] String类的特点 String在java.lang.String包中 1:特点 (1)String类型不能被继承,因为由final修饰 (2)Strin ...

  9. C语言祸根看不见的错误,那些年学指针从入门到入土都是内存问题

    是那样的爱学习 那一刻从入门到入土 丑拒 前言 C 语言程序中的内存错误非常有害:它们很常见,并且可能导致严重的后果,最难受的极大多数时候内存问题看不见,也摸不着.编译正确运行出错,让新手从入门到入土 ...

最新文章

  1. HDU-4454 Stealing a Cake 三分枚举
  2. 深圳腾讯内部Jetpack宝典意外流出!极致经典,堪称Android架构组件的天花板
  3. Big Data Exchange宣布在南京建设数据中心
  4. FastJson/spring boot: json输出
  5. 微服务架构基础之Service Mesh
  6. 网站视频解析 有的url资源放在浏览器能直接播放,有的却不行。
  7. ORA-23616:执行块5失败
  8. problem a: 简单的整数排序_python里的排序
  9. Java中字符串的学习(一)String类的概述及常见方法使用
  10. 管理SourceForge项目的方法
  11. git pull命令模式
  12. ubuntu20.04下开发海康威视网络摄像头sdk(二)云台基本控制(位姿控制)
  13. paip.输入法编程---带ord gudin去重复-
  14. catia孔深度符号标注_能回答出这39个问题,你就是SolidWorks高手
  15. 两种索引:MyISAM(非聚集索引)和InnoDB(聚集索引)的介绍
  16. android数据库降级_Android之sqlite数据库版本升级和降级的处理(onUpgrade和onDowngrade)...
  17. dell初始linux密码,常用设备管理口默认用户名密码汇总
  18. 【算法】独立任务最优调度问题
  19. Linux exec函数的使用
  20. JS逆向 | ob混淆一键还原工具

热门文章

  1. 卷积神经网络——第一周 卷积神经网络基础——第一部分
  2. c++中sort()的用法
  3. 深度学习基础 | RNN家族全面解析
  4. android window 半透明,popupWindow半透明背景
  5. linux改狗命令,linux安装安全狗
  6. python搜索网页特定区域内容_Python爬取练习:指定百度搜索的内容并提取网页的标题内容...
  7. linux adc是什么设备,Linux 驱动--ADC驱动
  8. 吴恩达深度学习之二《改善深层神经网络:超参数调试、正则化以及优化》学习笔记
  9. 吴恩达机器学习笔记3——线性代数
  10. 剑指offer (03):数组中重复的数字 (C++ Python 实现)