天天看點

15天玩轉redis —— 第二篇 基礎的字元串類型

我們都知道redis是采用c語言開發,那麼在c語言中表示string都是采用char[]數組的,然後你可能會想,那還不簡單,當我執行如下指令,肯定是直

接塞給char[]數組的。

15天玩轉redis —— 第二篇 基礎的字元串類型

如果你真的這麼想的話,會有幾個問題就要過來砍你了,先我們來找一個redis手冊,http://doc.redisfans.com/

15天玩轉redis —— 第二篇 基礎的字元串類型

第一:如果你每次都執行append函數,那是不是redis的char[]每次都需要再次擴容,這樣是不是每次都是耗時操作呢?

第二:如果你每次執行string中的strlen,那redis底層是不是每次都要周遊char數組來得到結果呢?

一: 探索redis中的string是如何存儲的

根據上面說的那些小情況,是以redis的作者沒有那麼傻,正常的邏輯應該是在char[]數組的層面上自己再來封裝一層,你說對不對???

1. sds結構體

在redis裡面是采用sds(simple dynamic string)來封裝char[]的,這個也是redis存儲的最小單元,下一個問題就是哪裡能看得到呢?我在wget壓

縮包的時候,裡面就有redis源碼啦,據說還隻有3w多行,這就告訴我們,有什麼問題,自己動手豐衣足食,對吧,為查找友善,我就把redis的源碼拖

到window上用vs打開,接下來我們看看sds長成啥樣???

15天玩轉redis —— 第二篇 基礎的字元串類型

可以看到它是定義在redis源碼中的sds.h源檔案中的,你可能會奇怪,這三個屬性是幹嘛用的???下面我簡單說一下。

<1> len: 标記char[]的長度, 有點類似我們c#中list的length一個意思。

<2> free: 标記char[]中未使用的元素個數,就是有幾個空坑的意思。

<3>buf[]:存放元素的坑,不一定和元素的實際個數相等,比如前面說的cnblogs。也有可能是[c][n][b][l][o][g][s][/0][][][]。

二:探索redis對象(redisobject)

前面說到的sds僅僅是char[]數組的封裝,并不能辨別redis中的5大類型,是以可想而知,redis還需要在sds上面進行封裝,是以就有了接下來的

redisobject對象,我們先看看它長成啥樣。

15天玩轉redis —— 第二篇 基礎的字元串類型

可以看到redisobject是在redis.h源代碼檔案中的,下面我簡單說說type和ptr屬性,詳細的東西在後續說。

<1> type 這個就是用來辨別redisobject是哪種類型,既然是哪種類型,肯定就有一個類型枚舉,對吧,肯定有了,給你看看。

15天玩轉redis —— 第二篇 基礎的字元串類型

<2> *ptr 可以看到這玩意還是個指針類型,它所指向的記憶體位址,你應該也知道了,就是所謂的sds枚舉類型。

好了,到現在你可以整合一下部落格開始處的:

針對上面的set指令,redis其實會建立兩個redisobject對象, 鍵的redisobject 和 值的redisojbect 其中它們的type=redis_string ,

也就都是字元串對象類型,其中的sds分别存儲的就是name和cnblogs的字元咯,好了,大概就這樣了。

三:挑選幾個有意思的指令

1. incr,incrby,decr,decrby

  這四個指令有點像c#中的interlocked類的方法,如果你了解interlocked,你應該就知道下面有各種原子自增,自減等等方法,如下圖:

15天玩轉redis —— 第二篇 基礎的字元串類型

redis這個自增有什麼好處呢?我覺得用這個生成訂單号還是蠻好的,我記得在攜程的時候,生成訂單号是專門的一個orderiddb中的func函數來生成的,

這樣orderid是不依賴于任何業務庫的,然後我們就可以相對友善的分庫分表了,現在用redis這樣做也挺好的。

15天玩轉redis —— 第二篇 基礎的字元串類型

其他的一些指令也沒什麼好說的了,大家可以對照redis手冊看一看就好了,就此結束,

繼續閱讀