本文分析Redis字符串的實(shí)現(xiàn)原理,內(nèi)容摘自新書(shū)《Redis核心原理與實(shí)踐》。這本書(shū)深入地分析了Redis常用特性的內(nèi)部機(jī)制與實(shí)現(xiàn)方式,內(nèi)容源自對(duì)Redis源碼的分析,并從中總結(jié)出設(shè)計(jì)思路、實(shí)現(xiàn)原理。通過(guò)閱讀本書(shū),讀者可以快速、輕松地了解Redis的內(nèi)部運(yùn)行機(jī)制。
Redis是一個(gè)鍵值對(duì)數(shù)據(jù)庫(kù)(key-value DB),下面是一個(gè)簡(jiǎn)單的Redis的命令:
> SET msg "hello wolrd"
該命令將鍵“msg”、值“hello wolrd”這兩個(gè)字符串保存到Redis數(shù)據(jù)庫(kù)中。
本章分析Redis如何在內(nèi)存中保存這些字符串。
Redis中的數(shù)據(jù)對(duì)象server.h/redisObject是Redis對(duì)內(nèi)部存儲(chǔ)的數(shù)據(jù)定義的抽象類(lèi)型,在深入分析Redis數(shù)據(jù)類(lèi)型前,我們先了解redisObject,它的定義如下:
typedef struct redisObject { unsigned type:4; unsigned encoding:4; unsigned lru:LRU_BITS; int refcount; void *ptr; } robj;
redisObject負(fù)責(zé)裝載Redis中的所有鍵和值。redisObject.ptr指向真正存儲(chǔ)數(shù)據(jù)的數(shù)據(jù)結(jié)構(gòu),redisObject .refcount、redisObject.lru等屬性則用于管理數(shù)據(jù)(數(shù)據(jù)共享、數(shù)據(jù)過(guò)期等)。
提示:type、encoding、lru使用了C語(yǔ)言中的位段定義,這3個(gè)屬性使用同一個(gè)unsigned int的不同bit位。這樣可以最大限度地節(jié)省內(nèi)存。
Redis定義了以下數(shù)據(jù)類(lèi)型和編碼,如表1-1所示。
本書(shū)第1部分會(huì)對(duì)表1-1中前五種數(shù)據(jù)類(lèi)型進(jìn)行分析,最后兩種數(shù)據(jù)類(lèi)型會(huì)在第5部分進(jìn)行分析。如果讀者現(xiàn)在對(duì)表1-1中內(nèi)容感到疑惑,則可以先帶著疑問(wèn)繼續(xù)閱讀本書(shū)。
我們知道,C語(yǔ)言中將空字符結(jié)尾的字符數(shù)組作為字符串,而Redis對(duì)此做了擴(kuò)展,定義了字符串類(lèi)型sds(Simple Dynamic String)。
Redis鍵都是字符串類(lèi)型,Redis中最簡(jiǎn)單的值類(lèi)型也是字符串類(lèi)型,
字符串類(lèi)型的Redis值可用于很多場(chǎng)景,如緩存HTML片段、記錄用戶(hù)登錄信息等。
提示:本節(jié)代碼如無(wú)特殊說(shuō)明,均在sds.h/sds.c中。
對(duì)于不同長(zhǎng)度的字符串,Redis定義了不同的sds結(jié)構(gòu)體:
typedef char *sds; struct __attribute__ ((__packed__)) sdshdr5 { unsigned char flags; char buf[]; }; struct __attribute__ ((__packed__)) sdshdr8 { uint8_t len; uint8_t alloc; unsigned char flags; char buf[]; }; ...
Redis還定義了sdshdr16、sdshdr32、sdshdr64結(jié)構(gòu)體。為了版面整潔,這里不展示sdshdr16、sdshdr32、sdshdr64 結(jié)構(gòu)體的代碼,它們與sdshdr8結(jié)構(gòu)體基本相同,只是len、alloc屬性使用了 uint16_t、uint32、uint64_t類(lèi)型。Redis定義不同sdshdr結(jié)構(gòu)體是為了針對(duì)不同長(zhǎng)度的字符串,使用合適的len、alloc屬性類(lèi)型,最大限度地節(jié)省內(nèi)存。
提示:sdshdr結(jié)構(gòu)體中的buf數(shù)組并沒(méi)有指定數(shù)組長(zhǎng)度,它是C99規(guī)范定義的柔性數(shù)組—結(jié)構(gòu)體中最后一個(gè)屬性可以被定義為一個(gè)大小可變的數(shù)組(該屬性前必須有其他屬性)。使用sizeof函數(shù)計(jì)算包含柔性數(shù)組的結(jié)構(gòu)體大小,返回結(jié)果不包括柔性數(shù)組占用的內(nèi)存。
另外,attribute((packed))關(guān)鍵字可以取消結(jié)構(gòu)體內(nèi)的字節(jié)對(duì)齊以節(jié)省內(nèi)存。
接下來(lái)看一下sds構(gòu)建函數(shù):
sds sdsnewlen(const void *init, size_t initlen) { void *sh; sds s; // [1] char type = sdsReqType(initlen); // [2] if (type == SDS_TYPE_5 initlen == 0) type = SDS_TYPE_8; // [3] int hdrlen = sdsHdrSize(type); unsigned char *fp; /* flags pointer. */ sh = s_malloc(hdrlen+initlen+1); ... // [4] s = (char*)sh+hdrlen; fp = ((unsigned char*)s)-1; switch(type) { case SDS_TYPE_5: { *fp = type | (initlen SDS_TYPE_BITS); break; } case SDS_TYPE_8: { SDS_HDR_VAR(8,s); sh->len = initlen; sh->alloc = initlen; *fp = type; break; } ... } if (initlen init) memcpy(s, init, initlen); s[initlen] = '\0'; // [5] return s; }
參數(shù)說(shuō)明:
【1】根據(jù)字符串長(zhǎng)度,判斷對(duì)應(yīng)的sdshdr類(lèi)型。
【2】長(zhǎng)度為0的字符串后續(xù)通常需要擴(kuò)容,不應(yīng)該使用sdshdr5,所以這里轉(zhuǎn)換為sdshdr8。
【3】sdsHdrSize函數(shù)負(fù)責(zé)查詢(xún)sdshdr結(jié)構(gòu)體的長(zhǎng)度,s_malloc函數(shù)負(fù)責(zé)申請(qǐng)內(nèi)存空間,申請(qǐng)的內(nèi)存空間長(zhǎng)度為hdrlen+initlen+1,其中hdrlen為sdshdr結(jié)構(gòu)體長(zhǎng)度(不包含buf屬性),initlen為字符串內(nèi)容長(zhǎng)度,最后一個(gè)字節(jié)用于存放空字符“\0”。s_malloc與C語(yǔ)言的malloc函數(shù)的作用相同,負(fù)責(zé)分配指定大小的內(nèi)存空間。
【4】給sdshdr屬性賦值。
SDS_HDR_VAR是一個(gè)宏,負(fù)責(zé)將sh指針轉(zhuǎn)化為對(duì)應(yīng)的sdshdr結(jié)構(gòu)體指針。
【5】注意,sds實(shí)際上就是char*的別名,這里返回的s指針指向sdshdr.buf屬性,即字符串內(nèi)容。Redis通過(guò)該指針可以直接讀/寫(xiě)字符串?dāng)?shù)據(jù)。
構(gòu)建一個(gè)內(nèi)容為“hello wolrd”的sds,其結(jié)構(gòu)如圖1-1所示。
sds的擴(kuò)容機(jī)制是一個(gè)很重要的功能。
sds sdsMakeRoomFor(sds s, size_t addlen) { void *sh, *newsh; // [1] size_t avail = sdsavail(s); size_t len, newlen; char type, oldtype = s[-1] SDS_TYPE_MASK; int hdrlen; if (avail >= addlen) return s; // [2] len = sdslen(s); sh = (char*)s-sdsHdrSize(oldtype); newlen = (len+addlen); // [3] if (newlen SDS_MAX_PREALLOC) newlen *= 2; else newlen += SDS_MAX_PREALLOC; // [4] type = sdsReqType(newlen); if (type == SDS_TYPE_5) type = SDS_TYPE_8; // [5] hdrlen = sdsHdrSize(type); if (oldtype==type) { newsh = s_realloc(sh, hdrlen+newlen+1); if (newsh == NULL) return NULL; s = (char*)newsh+hdrlen; } else { newsh = s_malloc(hdrlen+newlen+1); if (newsh == NULL) return NULL; memcpy((char*)newsh+hdrlen, s, len+1); s_free(sh); s = (char*)newsh+hdrlen; s[-1] = type; sdssetlen(s, len); } // [6] sdssetalloc(s, newlen); return s; }
參數(shù)說(shuō)明:
addlen:要求擴(kuò)容后可用長(zhǎng)度(alloc-len)大于該參數(shù)。
【1】獲取當(dāng)前可用空間長(zhǎng)度。如果當(dāng)前可用空間長(zhǎng)度滿(mǎn)足要求,則直接返回。
【2】sdslen負(fù)責(zé)獲取字符串長(zhǎng)度,由于sds.len中記錄了字符串長(zhǎng)度,該操作復(fù)雜度為O(1)。這里len變量為原sds字符串長(zhǎng)度,newlen變量為新sds長(zhǎng)度。sh指向原sds的sdshdr結(jié)構(gòu)體。
【3】預(yù)分配比參數(shù)要求多的內(nèi)存空間,避免每次擴(kuò)容都要進(jìn)行內(nèi)存拷貝操作。新sds長(zhǎng)度如果小于SDS_MAX_PREALLOC(默認(rèn)為1024×1024,單位為字節(jié)),則新sds長(zhǎng)度自動(dòng)擴(kuò)容為2倍。否則,新sds長(zhǎng)度自動(dòng)增加SDS_MAX_PREALLOC。
【4】sdsReqType(newlen)負(fù)責(zé)計(jì)算新的sdshdr類(lèi)型。注意,擴(kuò)容后的類(lèi)型不使用sdshdr5,該類(lèi)型不支持?jǐn)U容操作。
【5】如果擴(kuò)容后sds還是同一類(lèi)型,則使用s_realloc函數(shù)申請(qǐng)內(nèi)存。否則,由于sds結(jié)構(gòu)已經(jīng)變動(dòng),必須移動(dòng)整個(gè)sds,直接分配新的內(nèi)存空間,并將原來(lái)的字符串內(nèi)容復(fù)制到新的內(nèi)存空間。s_realloc與C語(yǔ)言realloc函數(shù)的作用相同,負(fù)責(zé)為給定指針重新分配給定大小的內(nèi)存空間。它會(huì)嘗試在給定指針原地址空間上重新分配,如原地址空間無(wú)法滿(mǎn)足要求,則分配新內(nèi)存空間并復(fù)制內(nèi)容。
【6】更新sdshdr.alloc屬性。
對(duì)上面“hello wolrd”的sds調(diào)用sdsMakeRoomFor(sds,64),則生成的sds如圖1-2所示。
從圖1-2中可以看到,使用len記錄字符串長(zhǎng)度后,字符串中可以存放空字符。Redis字符串支持二進(jìn)制安全,可以將用戶(hù)的輸入存儲(chǔ)為沒(méi)有任何特定格式意義的原始數(shù)據(jù)流,因此Redis字符串可以存儲(chǔ)任何數(shù)據(jù),比如圖片數(shù)據(jù)流或序列化對(duì)象。C語(yǔ)言字符串將空字符作為字符串結(jié)尾的特定標(biāo)記字符,它不是二進(jìn)制安全的。
sds常用函數(shù)如表1-2所示。
函數(shù) | 作用 |
---|---|
sdsnew,sdsempty | 創(chuàng)建sds |
sdsfree,sdsclear,sdsRemoveFreeSpace | 釋放sds,清空sds中的字符串內(nèi)容,移除sds剩余的可用空間 |
sdslen | 獲取sds字符串長(zhǎng)度 |
sdsdup | 將給定字符串復(fù)制到sds中,覆蓋原字符串 |
sdscat | 將給定字符串拼接到sds字符串內(nèi)容后 |
sdscmp | 對(duì)比兩個(gè)sds字符串是否相同 |
sdsrange | 獲取子字符串,不在指定范圍內(nèi)的字符串將被清除 |
字符串類(lèi)型一共有3種編碼:
在該編碼中,redisObject、sds結(jié)構(gòu)存放在一塊連續(xù)內(nèi)存塊中,如圖1-3所示。
OBJ_ENCODING_EMBSTR編碼是Redis針對(duì)短字符串的優(yōu)化,有如下優(yōu)點(diǎn):
(1)內(nèi)存申請(qǐng)和釋放都只需要調(diào)用一次內(nèi)存操作函數(shù)。
(2)redisObject、sdshdr結(jié)構(gòu)保存在一塊連續(xù)的內(nèi)存中,減少了內(nèi)存碎片。
我們向Redis發(fā)送一個(gè)請(qǐng)求后,Redis會(huì)解析請(qǐng)求報(bào)文,并將命令、參數(shù)轉(zhuǎn)化為redisObjec。
object.c/createStringObject函數(shù)負(fù)責(zé)完成該操作:
robj *createStringObject(const char *ptr, size_t len) { if (len = OBJ_ENCODING_EMBSTR_SIZE_LIMIT) return createEmbeddedStringObject(ptr,len); else return createRawStringObject(ptr,len); }
可以看到,這里根據(jù)字符串長(zhǎng)度,將encoding轉(zhuǎn)化為OBJ_ENCODING_RAW或OBJ_ENCODING_EMBSTR的redisObject。
將參數(shù)轉(zhuǎn)換為redisObject后,Redis再將redisObject存入數(shù)據(jù)庫(kù),例如:
> SET Introduction "Redis is an open source (BSD licensed), in-memory data structure store, used as a database, cache and message broker. "
Redis會(huì)將鍵“Introduction”、值“Redis...”轉(zhuǎn)換為兩個(gè)redisObject,再將redisObject存入數(shù)據(jù)庫(kù),結(jié)果如圖1-4所示。
Redis中的鍵都是字符串類(lèi)型,并使用OBJ_ENCODING_RAW、OBJ_ENCODING_ EMBSTR編碼,而Redis還會(huì)嘗試將字符串類(lèi)型的值轉(zhuǎn)換為OBJ_ENCODING_INT 編碼。object.c/tryObjectEncoding函數(shù)完成該操作:
robj *tryObjectEncoding(robj *o) { long value; sds s = o->ptr; size_t len; ... // [1] if (o->refcount > 1) return o; len = sdslen(s); // [2] if (len = 20 string2l(s,len,value)) { // [3] if ((server.maxmemory == 0 || !(server.maxmemory_policy MAXMEMORY_FLAG_NO_SHARED_INTEGERS)) value >= 0 value OBJ_SHARED_INTEGERS) { decrRefCount(o); incrRefCount(shared.integers[value]); return shared.integers[value]; } else { // [4] if (o->encoding == OBJ_ENCODING_RAW) { sdsfree(o->ptr); o->encoding = OBJ_ENCODING_INT; o->ptr = (void*) value; return o; } else if (o->encoding == OBJ_ENCODING_EMBSTR) { // [5] decrRefCount(o); return createStringObjectFromLongLongForValue(value); } } } // [6] if (len = OBJ_ENCODING_EMBSTR_SIZE_LIMIT) { robj *emb; if (o->encoding == OBJ_ENCODING_EMBSTR) return o; emb = createEmbeddedStringObject(s,sdslen(s)); decrRefCount(o); return emb; } // [7] trimStringObjectIfNeeded(o); return o; }
【1】該數(shù)據(jù)對(duì)象被多處引用,不能再進(jìn)行編碼操作,否則會(huì)影響其他地方的正常運(yùn)行。
【2】如果字符串長(zhǎng)度小于或等于20,則調(diào)用string2l函數(shù)嘗試將其轉(zhuǎn)換為long long類(lèi)型,如果成功則返回1。
在C語(yǔ)言中,long long占用8字節(jié),取值范圍是-9223372036854775808~9223372036854775807,因此最多能保存長(zhǎng)度為19的字符串轉(zhuǎn)換后的數(shù)值,加上負(fù)數(shù)的符號(hào)位,一共20位。
下面是字符串可以轉(zhuǎn)換為OBJ_ENCODING_INT 編碼的處理步驟。
【3】首先嘗試使用shared.integers中的共享數(shù)據(jù),避免重復(fù)創(chuàng)建相同數(shù)據(jù)對(duì)象而浪費(fèi)內(nèi)存。shared是Redis啟動(dòng)時(shí)創(chuàng)建的共享數(shù)據(jù)集,存放了Redis中常用的共享數(shù)據(jù)。shared.integers是一個(gè)整數(shù)數(shù)組,存放了小數(shù)字0~9999,共享于各個(gè)使用場(chǎng)景。
注意:如果配置了server.maxmemory,并使用了不支持共享數(shù)據(jù)的淘汰算法(LRU、LFU),那么這里不能使用共享數(shù)據(jù),因?yàn)檫@時(shí)每個(gè)數(shù)據(jù)中都必須存在一個(gè)redisObjec.lru屬性,這些算法才可以正常工作。
【4】如果不能使用共享數(shù)據(jù)并且原編碼格式為OBJ_ENCODING_RAW,則將redisObject.ptr原來(lái)的sds類(lèi)型替換為字符串轉(zhuǎn)換后的數(shù)值。
【5】如果不能使用共享數(shù)據(jù)并且原編碼格式為OBJ_ENCODING_EMBSTR,由于redisObject、sds存放在同一個(gè)內(nèi)存塊中,無(wú)法直接替換redisObject.ptr,所以調(diào)用createString- ObjectFromLongLongForValue函數(shù)創(chuàng)建一個(gè)新的redisObject,編碼為OBJ_ENCODING_INT,redisObject.ptr指向long long類(lèi)型或long類(lèi)型。
【6】到這里,說(shuō)明字符串不能轉(zhuǎn)換為OBJ_ENCODING_INT 編碼,嘗試將其轉(zhuǎn)換為OBJ_ENCODING_EMBSTR編碼。
【7】到這里,說(shuō)明字符串只能使用OBJ_ENCODING_RAW編碼,嘗試釋放sds中剩余的可用空間。
字符串類(lèi)型的實(shí)現(xiàn)代碼在t_string.c中,讀者可以查看源碼了解更多實(shí)現(xiàn)細(xì)節(jié)。
提示:server.c/redisCommandTable定義了每個(gè)Redis命令與對(duì)應(yīng)的處理函數(shù),讀者可以從這里查找感興趣的命令的處理函數(shù)。
struct redisCommand redisCommandTable[] = { ... {"get",getCommand,2, "read-only fast @string", 0,NULL,1,1,1,0,0,0}, {"set",setCommand,-3, "write use-memory @string", 0,NULL,1,1,1,0,0,0}, ... }
GET命令的處理函數(shù)為getCommand,SET命令的處理函數(shù)為setCommand,以此類(lèi)推。
另外,我們可以通過(guò)TYPE命令查看數(shù)據(jù)對(duì)象類(lèi)型,通過(guò)OBJECT ENCODING命令查看編碼:
> SET msg "hello world" OK > TYPE msg string > OBJECT ENCODING msg "embstr" > SET Introduction "Redis is an open source (BSD licensed), in-memory data structure store, used as a database, cache and message broker. " OK > TYPE Introduction string > OBJECT ENCODING info "raw" > SET page 1 OK > TYPE page string > OBJECT ENCODING page "int"
Redis中的所有鍵和值都是redisObject變量。
本文內(nèi)容摘自作者新書(shū)《Redis核心原理與實(shí)踐》,這本書(shū)深入地分析了Redis常用特性的內(nèi)部機(jī)制與實(shí)現(xiàn)方式,大部分內(nèi)容源自對(duì)Redis源碼的分析,并從中總結(jié)出設(shè)計(jì)思路、實(shí)現(xiàn)原理。通過(guò)閱讀本書(shū),讀者可以快速、輕松地了解Redis的內(nèi)部運(yùn)行機(jī)制。
京東鏈接
豆瓣鏈接
到此這篇關(guān)于Redis核心原理與實(shí)踐之字符串實(shí)現(xiàn)原理的文章就介紹到這了,更多相關(guān)Redis字符串實(shí)現(xiàn)原理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
標(biāo)簽:北京 吉安 大慶 江蘇 楊凌 果洛 朝陽(yáng) 臺(tái)州
巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Redis核心原理與實(shí)踐之字符串實(shí)現(xiàn)原理》,本文關(guān)鍵詞 Redis,核心,原理,與,實(shí)踐,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。