成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Redis概念以及底層數(shù)據(jù)結(jié)構(gòu)

數(shù)據(jù)庫 其他數(shù)據(jù)庫 Redis
Redis是一個開源的使用ANSI C語言編寫、遵守BSD協(xié)議、支持網(wǎng)絡(luò)、可基于內(nèi)存亦可持久化的日志型、Key-Value數(shù)據(jù)庫,并提供多種語言的API。

Redis 簡介

REmote DIctionary Server(Redis) 是一個由SalvatoreSanfilippo寫的key-value存儲系統(tǒng)。

Redis是一個開源的使用ANSI C語言編寫、遵守BSD協(xié)議、支持網(wǎng)絡(luò)、可基于內(nèi)存亦可持久化的日志型、Key-Value數(shù)據(jù)庫,并提供多種語言的API。

它通常被稱為數(shù)據(jù)結(jié)構(gòu)服務(wù)器,因?yàn)橹?value)可以是字符串(String), 哈希(Map), 列表(list), 集合(sets) 和有序集合(sorted sets)等類型。

Redis特點(diǎn)

Redis 是完全開源免費(fèi)的,遵守BSD協(xié)議,是一個高性能的key-value數(shù)據(jù)庫。

Redis 與其他 key - value 緩存產(chǎn)品有以下三個特點(diǎn):

  • Redis支持?jǐn)?shù)據(jù)的持久化,可以將內(nèi)存中的數(shù)據(jù)保持在磁盤中,重啟的時候可以再次加載進(jìn)行使用。
  • Redis不僅僅支持簡單的key-value類型的數(shù)據(jù),同時還提供list,set,zset,hash等數(shù)據(jù)結(jié)構(gòu)的存儲。
  • Redis支持?jǐn)?shù)據(jù)的備份,即master-slave模式的數(shù)據(jù)備份。

Redis 優(yōu)勢

性能極高 – Redis能讀的速度是110000次/s,寫的速度是81000次/s 。

豐富的數(shù)據(jù)類型 – Redis支持 Strings, Lists, Hashes, Sets 及 Ordered Sets 數(shù)據(jù)類型操作。

原子 – Redis的所有操作都是原子性的,同時Redis還支持對幾個操作全并后的原子性執(zhí)行。

豐富的特性 – Redis 還支持 publish/subscribe, 隊(duì)列,key 過期等等特性。

Redis對象類型簡介

Redis是一種key/value型數(shù)據(jù)庫,其中,每個key和value都是使用對象表示的。

比如,我們執(zhí)行以下代碼: 

  1. redis> SET message "hello redis" 

其中的key是message,是一個包含了字符串"message"的對象。而value是一個包含了"hello redis"的對象。

Redis共有五種對象的類型,分別是:

類型常量 對象的名稱
REDIS_STRING 字符串對象
REDIS_LIST 列表對象
REDIS_HASH 哈希對象
REDIS_SET 集合對象
REDIS_ZSET 有序集合對象

Redis中的一個對象的結(jié)構(gòu)體表示如下: 

  1. typedef struct redisObject {  
  2. // 類型  
  3. unsigned type:4;  
  4. // 編碼方式  
  5. unsigned encoding: 4;  
  6. // 引用計(jì)數(shù)  
  7. int refcount;  
  8. // 指向?qū)ο蟮闹?nbsp; 
  9. void *ptr;  
  10. } robj;  

type表示了該對象的對象類型,即上面五個中的一個。但為了提高存儲效率與程序執(zhí)行效率,每種對象的底層數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)都可能不止一種。encoding就表示了對象底層所使用的編碼。

  • Redis對象底層數(shù)據(jù)結(jié)構(gòu)
編碼常量 編碼所對應(yīng)的底層數(shù)據(jù)結(jié)構(gòu)
REDIS_ENCODING_INT long 類型的整數(shù)
REDIS_ENCODING_EMBSTR embstr 編碼的簡單動態(tài)字符串
REDIS_ENCODING_RAW 簡單動態(tài)字符串
REDIS_ENCODING_HT 字典
REDIS_ENCODING_LINKEDLIST 雙端鏈表
REDIS_ENCODING_ZIPLIST 壓縮列表
REDIS_ENCODING_INTSET 整數(shù)集合
REDIS_ENCODING_SKIPLIST 跳躍表和字典
  • 字符串對象

字符串對象的編碼可以是int、raw或者embstr

如果一個字符串的內(nèi)容可以轉(zhuǎn)換為long,那么該字符串就會被轉(zhuǎn)換成為long類型,對象的ptr就會指向該long,并且對象類型也用int類型表示。

普通的字符串有兩種,embstr和raw。embstr應(yīng)該是Redis 3.0新增的數(shù)據(jù)結(jié)構(gòu),在2.8中是沒有的。如果字符串對象的長度小于39字節(jié),就用embstr對象。否則用傳統(tǒng)的raw對象。 

  1. #define REDIS_ENCODING_EMBSTR_SIZE_LIMIT 44  
  2. robj *createStringObject(char *ptr, size_t len) {  
  3. if (len <= REDIS_ENCODING_EMBSTR_SIZE_LIMIT)  
  4. return createEmbeddedStringObject(ptr,len);  
  5. else  
  6. return createRawStringObject(ptr,len);  
  7.  

embstr的好處有如下幾點(diǎn):

  1. embstr的創(chuàng)建只需分配一次內(nèi)存,而raw為兩次(一次為sds分配對象,另一次為objet分配對象,embstr省去了***次)。
  2. 相對地,釋放內(nèi)存的次數(shù)也由兩次變?yōu)橐淮巍?/li>
  3. embstr的objet和sds放在一起,更好地利用緩存帶來的優(yōu)勢。

raw和embstr的區(qū)別可以用下面兩幅圖所示:

 

  • 列表對象

列表對象的編碼可以是ziplist或者linkedlist

  1. ziplist是一種壓縮鏈表,它的好處是更能節(jié)省內(nèi)存空間,因?yàn)樗鎯Φ膬?nèi)容都是在連續(xù)的內(nèi)存區(qū)域當(dāng)中的。當(dāng)列表對象元素不大,每個元素也不大的時候,就采用ziplist存儲但當(dāng)數(shù)據(jù)量過大時就ziplist就不是那么好用了。因?yàn)闉榱吮WC他存儲內(nèi)容在內(nèi)存中的連續(xù)性,插入的復(fù)雜度是O(N),即每次插入都會重新進(jìn)行realloc。如下圖所示,對象結(jié)構(gòu)中ptr所指向的就是一個ziplist整個ziplist只需要malloc一次,它們在內(nèi)存中是一塊連續(xù)的區(qū)域。

 

linkedlist是一種雙向鏈表。它的結(jié)構(gòu)比較簡單,節(jié)點(diǎn)中存放pre和next兩個指針,還有節(jié)點(diǎn)相關(guān)的信息。當(dāng)每增加一個node的時候,就需要重新malloc一塊內(nèi)存。

 

  • 哈希對象

哈希對象的底層實(shí)現(xiàn)可以是ziplist或者h(yuǎn)ashtable。

ziplist中的哈希對象是按照key1,value1,key2,value2這樣的順序存放來存儲的。當(dāng)對象數(shù)目不多且內(nèi)容不大時,這種方式效率是很高的。

hashtable的是由dict這個結(jié)構(gòu)來實(shí)現(xiàn)的, dict是一個字典,其中的指針dicht ht[2] 指向了兩個哈希表 

  1. typedef struct dict {  
  2. dictType *type;  
  3. void *privdata;  
  4. dictht ht[2];  
  5. long rehashidx; /* rehashing not in progress if rehashidx == -1 */  
  6. int iterators; /* number of iterators currently running */  
  7. } dict;  
  8. typedef struct dictht {  
  9. dictEntry **table;  
  10. unsigned long size;  
  11. unsigned long sizemask;  
  12. unsigned long used;  
  13. } dictht;  

dicht[0] 是用于真正存放數(shù)據(jù),dicht[1]一般在哈希表元素過多進(jìn)行rehash的時候用于中轉(zhuǎn)數(shù)據(jù)。

dictht中的table用語真正存放元素了,每個key/value對用一個dictEntry表示,放在dictEntry數(shù)組中。

 

  • 集合對象

集合對象的編碼可以是intset或者h(yuǎn)ashtable

intset是一個整數(shù)集合,里面存的為某種同一類型的整數(shù),支持如下三種長度的整數(shù): 

  1. #define INTSET_ENC_INT16 (sizeof(int16_t))  
  2. #define INTSET_ENC_INT32 (sizeof(int32_t))  
  3. #define INTSET_ENC_INT64 (sizeof(int64_t))  

intset是一個有序集合,查找元素的復(fù)雜度為O(logN),但插入時不一定為O(logN),因?yàn)橛锌赡苌婕暗缴壊僮?。比如?dāng)集合里全是int16_t型的整數(shù),這時要插入一個int32_t,那么為了維持集合中數(shù)據(jù)類型的一致,那么所有的數(shù)據(jù)都會被轉(zhuǎn)換成int32_t類型,涉及到內(nèi)存的重新分配,這時插入的復(fù)雜度就為O(N)了。

intset不支持降級操作。

  • 有序集合對象

有序集合的編碼可能兩種,一種是ziplist,另一種是skiplist與dict的結(jié)合。

ziplist作為集合和作為哈希對象是一樣的,member和score順序存放。按照score從小到大順序排列

skiplist是一種跳躍表,它實(shí)現(xiàn)了有序集合中的快速查找,在大多數(shù)情況下它的速度都可以和平衡樹差不多。但它的實(shí)現(xiàn)比較簡單,可以作為平衡樹的替代品。它的結(jié)構(gòu)比較特殊。下面分別是跳躍表skiplist和它內(nèi)部的節(jié)點(diǎn)skiplistNode的結(jié)構(gòu)體: 

  1. /*  
  2. * 跳躍表  
  3. */  
  4. typedef struct zskiplist {  
  5. // 頭節(jié)點(diǎn),尾節(jié)點(diǎn)  
  6. struct zskiplistNode *header, *tail;  
  7. // 節(jié)點(diǎn)數(shù)量  
  8. unsigned long length;  
  9. // 目前表內(nèi)節(jié)點(diǎn)的***層數(shù)  
  10. int level;  
  11. } zskiplist;  
  12. /* ZSETs use a specialized version of Skiplists */  
  13. /*  
  14. * 跳躍表節(jié)點(diǎn)  
  15. */  
  16. typedef struct zskiplistNode {  
  17. // member 對象  
  18. robj *obj;  
  19. // 分值  
  20. double score;  
  21. // 后退指針  
  22. struct zskiplistNode *backward;  
  23. // 層  
  24. struct zskiplistLevel {  
  25. // 前進(jìn)指針  
  26. struct zskiplistNode *forward;  
  27. // 這個層跨越的節(jié)點(diǎn)數(shù)量  
  28. unsigned int span;  
  29. } level[];  
  30. } zskiplistNode;  

head和tail分別指向頭節(jié)點(diǎn)和尾節(jié)點(diǎn),然后每個skiplistNode里面的結(jié)構(gòu)又是分層的(即level數(shù)組)

用圖表示,大概是下面這個樣子:

 

總結(jié)

以上簡單介紹了Redis的簡介,特性以及五種對象類型和五種對象類型的底層實(shí)現(xiàn)。事實(shí)上,Redis的高效性和靈活性正是得益于同一個對象類型采用不同的底層結(jié)構(gòu),并且在必要的時候?qū)Χ哌M(jìn)行轉(zhuǎn)換,還有就是各種底層結(jié)構(gòu)對內(nèi)存的合理利用。

責(zé)任編輯:龐桂玉 來源: segmentfault
相關(guān)推薦

2023-09-15 08:14:48

HashMap負(fù)載因子

2019-10-29 08:59:16

Redis底層數(shù)據(jù)

2022-05-23 08:19:19

Redis數(shù)據(jù)結(jié)構(gòu)內(nèi)存

2023-04-28 08:53:09

2023-01-09 08:42:04

String數(shù)據(jù)類型

2023-06-08 07:25:56

數(shù)據(jù)庫索引數(shù)據(jù)結(jié)構(gòu)

2019-06-12 22:51:57

Redis軟件開發(fā)

2020-05-20 09:55:42

Git底層數(shù)據(jù)

2023-10-31 08:51:25

數(shù)據(jù)結(jié)構(gòu)存儲數(shù)據(jù)

2025-01-15 12:20:41

2025-01-14 08:00:00

RedisList數(shù)據(jù)結(jié)構(gòu)

2023-11-12 21:49:10

Redis數(shù)據(jù)庫

2019-06-21 15:20:05

Redis數(shù)據(jù)結(jié)構(gòu)數(shù)據(jù)庫

2020-03-20 10:47:51

Redis數(shù)據(jù)庫字符串

2023-03-06 08:40:43

RedisListJava

2021-08-29 07:41:48

數(shù)據(jù)HashMap底層

2020-07-07 07:34:29

RedisSDS數(shù)據(jù)結(jié)構(gòu)

2020-12-31 05:31:01

數(shù)據(jù)結(jié)構(gòu)算法

2010-06-09 15:04:12

2021-08-31 07:36:22

LinkedListAndroid數(shù)據(jù)結(jié)構(gòu)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 一区在线观看 | 精品国产一区二区三区av片 | 欧美一级一区 | 懂色一区二区三区免费观看 | 久久久久久国产精品mv | 日韩国产一区二区三区 | 国产综合在线视频 | 欧美日本一区二区 | 久久久精品一区二区 | caoporn国产精品免费公开 | xxx.在线观看| av网站免费| 草久久 | 久久精品一区 | 午夜视频在线播放 | 热99视频| av网站免费观看 | 在线观看亚 | 男人的天堂在线视频 | 天天干 夜夜操 | 亚洲自拍一区在线观看 | 亚洲精品一区二区三区免 | 特级特黄特色的免费大片 | 久久天堂| 久久久久久成人 | 国产精品视频区 | 中文成人无字幕乱码精品 | 成人午夜在线 | 久草成人 | 九九久久免费视频 | 国产高潮好爽受不了了夜色 | 九一视频在线观看 | 国内自拍视频在线观看 | 91精品久久久久久久久 | 欧美亚洲国产日韩 | 国产成人一区在线 | 福利片在线看 | 亚洲a在线观看 | 成人在线免费观看视频 | 成人在线小视频 | 国产一区二区精品在线观看 |