设为首页 收藏本站
查看: 1347|回复: 0

[经验分享] Redis数据类型之ZSet(五)

[复制链接]

尚未签到

发表于 2017-12-21 23:39:17 | 显示全部楼层 |阅读模式
  前言:有序集合zset跟其他类型一样,同样有几种编码方式。主要有两种编码方式,REDIS_ENCODING_ZIPLIST和REDIS_ENCODING_SKIPLIST。ziplist可以表示较小的有序集合, skiplist表示任意大小的有序集合。
  何时用REDIS_ENCODING_ZIPLIST?
  (1)zset会根据zadd命令添加的第一个元素的长度大小来选择编码方式,满足zset_max_ziplist_entries的值不为0,第一个元素的长度小于server.zset_max_ziplist_value,否则就使用skiplist。
  (2)当待新加的新的字符串长度超过zset_max_ziplist_value(默认值64)时或者ziplist保存的节点数量超过server.zset_max_ziplist_entries(默认值128)时使用skiplist。
  相关代码一目了然:创建ziplist前面有介绍,在这里不再敖述相关代码在object.c和ziplist.c中。
  

  if (zobj == NULL) {if (xx) goto reply_to_client; /* No key + XX option: nothing to do. */  if (server.zset_max_ziplist_entries == 0 ||
  server.zset_max_ziplist_value < sdslen(c->argv[scoreidx+1]->ptr))
  {
  zobj = createZsetObject();
  } else {
  zobj = createZsetZiplistObject();
  }
  dbAdd(c->db,key,zobj);
  } else {
  if (zobj->type != OBJ_ZSET) {
  addReply(c,shared.wrongtypeerr);
  goto cleanup;
  }
  }
  

  Skiplist:跳跃表是一种随机化的数据结构,基于链表,其效率可以比拟平衡二叉树,插入、查找、删除都可以在对数期望时间内完成(平均复杂度为O(logN),最坏为O(N)),但是又比平衡树简单直观。
  维基百科提供的Skiplist图:
   DSC0000.png
  主要有四部分组成
  A:head:节点指针
  B:node:元素值,每个节点有一层或多层
  C:level: 指向该层下一个节点的指针
  D:tail:全部为null  
  Redis的Skiplist代码如下:
  

/* ZSETs use a specialized version of Skiplists */  
typedef
struct zskiplistNode {  robj
*obj;double score;struct zskiplistNode *backward;struct zskiplistLevel {struct zskiplistNode *forward;  unsigned
int span;  } level[];
  
} zskiplistNode;
  
typedef
struct zskiplist {struct zskiplistNode *header, *tail;  unsigned
long length;int level;  
} zskiplist;
  

  
typedef
struct zset {  dict
*dict;  zskiplist
*zsl;  
} zset;
  

  可以清楚的看到redis的skiplist里面多了dict,forward,span几个元素,这三个元素有什么好处呢?
  1、dict维护了skiplist的元素值(key)和分数(value)用于快读的查找元素对应的分值以及判断元素是否存在。
  2、forward前向指针,用于从底层表尾向表头方向遍历。的结构中存在span跨度字段
  3、span存在于forward中,这个跨度字段的出现有助于快速计算元素在整个集合中的排名
  另外redis的跳跃表中,允许重复的score出现,多个不同的元素score值可以相同,若score值相同时,需要对比member,按字典排序存储在跳表结构中。
  相关操作:
  ZADD key score member [[score member] [score member] ...]
  将一个或多个元素以及值加入到有序集中。不存在创建,存在更新。
  ZREM key member [member ...]
  移除一个或这个成员,不存在忽略
  ZCARD key
  返回有序集的基数,不存在返回0
  ZCOUNT key min max
  返回score在min和max之间的成员
  ZSCORE key member
  返回有序集中,成员的score值
  ZINCRBY key increment member
  为有序集的成员的score加上相应的增量,不存在则创建
  ZRANGE key start stop
  根据基数返回有序集key中,指定区间内的成员,score值从小到大排列
  ZREVRANGE key start stop
  根据基数返回有序集key中,指定区间内的成员,score值从大到小排列
  ZRANGEBYSCORE key min max [WITHSCORES] [LIMIT offset count]
  返回有序集key中,所有score值介于min和max之间的成员。有序集成员按score值从小到大排列
  ZREVRANGEBYSCORE key max min [WITHSCORES] [LIMIT offset count]
  返回有序集key中,所有score值介于min和max之间的成员。有序集成员按score值从大到小排列
  ZRANK key member
  返回有序集key中成员member的排名。有序集成员按score值从小到大排列
  ZREVRANK key member
  返回有序集key中成员member的排名。有序集成员按score值从大到小排列
  ZREMRANGEBYRANK key start stop
  移除有序集key中,指定排名(rank)区间内的所有成员。
  ZREMRANGEBYSCORE key min max
  移除有序集key中,所有score值介于min和max之间的成员。
  ZINTERSTORE destination numkeys key [key ...] [WEIGHTS weight [weight ...]] [AGGREGATE SUM|MIN|MAX]
  计算一个或多个有序集的交集,其中给定的key必须以numkeys参数指定,并将该交集储存到destination
  注:结果集中某个成员的score值是所有给定集下该成员score值之和
  ZUNIONSTORE destination numkeys key [key ...] [WEIGHTS weight [weight ...]] [AGGREGATE SUM|MIN|MAX]
  算给定的一个或多个有序集的并集,其中给定key的数量必须以numkeys参数指定,并将该并集储存到destination
  注:结果集中某个成员的score值是所有给定集
  以上的min和max均包含等于。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-426675-1-1.html 上篇帖子: 转自:http://www.jianshu.com/p/af7043e6c8f9 下篇帖子: Redis的服务命令(实现开机自启动)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表