每個數據庫都有本身的數據類型。一樣子redis爲咱們提供了五種類型的數據——字符串、哈希、列表、集合、有序集合。咱們知道關係型數據的數據存放型式是一張二維表。用行和列來表示數據之間的關係。redis是一個nosql數據庫固然不可能在用什麼二維表的形式來表示了。他全部的數據都是以key=value的形式來存放的。每一種數據類型都有數據結構和內部編碼的概念。數據結構大家能夠理解他們的存放時的結構。而內部編碼就是數據結構的具體實現。可是一種數據結構能夠對應多種內部編碼的實現。接下來筆者就要去看看每一種數據類型相關的操做命令和數據結構,內部編碼。redis
字符串類型,筆者認爲比較簡單的類型。sql
數據結果:一串值數據庫
內部編碼:一共有三種:int embstr raw網絡
例子:數據結構
127.0.0.1:6379> set k1 1002 OK 127.0.0.1:6379> set k2 "i am aomi" OK 127.0.0.1:6379> set k3 "i am aomi i am aomi i am aomi i am aomi i am aomi i am aomi i am aomi i am aomi i am aomi i am aomi i am aomi " OK 127.0.0.1:6379> object encoding k3 "raw" 127.0.0.1:6379> object encoding k2 "embstr" 127.0.0.1:6379> object encoding k1 "int" 127.0.0.1:6379> keys * 1) "k3" 2) "k1" 3) "k2" 127.0.0.1:6379> get k2 "i am aomi"
筆者向redis時面存放了三個值。他這三個值對應的鍵爲:k一、k二、k3。nosql
語法:學習
SET key value [EX seconds] [PX milliseconds] [NX|XX]
key:鍵名ui
value:鍵對應的值編碼
EX:表示有效的時間。0表示永存,大於0表示在幾秒內有效。code
PX:表示有效時間。只是單位是毫秒。
NX:表示不存在的時候才能夠增長成功。
XX:表示只有存在的時候才能夠增長成功。
語法:
OBJECT subcommand [arguments [arguments]]
subcommand:這個有三個值 refcount(用於查看對象引用次數)、encoding(查看內部編碼)、idletime(存在的時間)。
語法:
KEYS pattern
pattern:*表示所有。*aaa表示查找以aaa結尾的。[a,b]123表示查看以a或是b,而且後面是123。相信不用筆者都說明了。
語法:
GET key
key:鍵的名稱
學下語法以後,咱們能夠從上面看到字串符有三種內部編碼了吧。若是你輸入是一個數字的話,通常都是int。若是輸入不是一個數字的話,是embstr。若是輸入的字符串長度大於512的話。就會變成raw
筆者來一個設置有效時間的數據吧。
127.0.0.1:6379> set k5 v5 ex 10 OK 127.0.0.1:6379> ttl k5 (integer) 4 127.0.0.1:6379> ttl k5 (integer) -2
上面ttl用於查看當前鍵是的有效時間。
上面的都是一個一個增長有沒有一次增長多個呢?固然是有的。
127.0.0.1:6379> mset k1 v1 k2 v2 k3 v3 OK 127.0.0.1:6379> mget k1 k2 k3 1) "v1" 2) "v2" 3) "v3" 127.0.0.1:6379>
事實上學習redis的命令是一件很簡單的事情。筆者是一邊看命令手冊一邊寫命令的。因此你們也能夠這樣子。本身寫過一遍基本上都不會忘記。若是有不懂的話,查看一下手冊就能夠了。至於手冊網絡上不少。建義能夠去官網看看。
這個數據類型算是這五種類型中最爲複雜的。存放形式不用說key=value。只是這個value裏面就不同子。是一個field=value形式的數據值。
數據結構:key: value(field=value)。不知道筆者這樣子表示大家看得懂多。
內部編碼:一共有倆種。一種是ziplist,二種是hashtable。
列子:
127.0.0.1:6379> hset user:1 name aomi (integer) 1 127.0.0.1:6379> hset user:1 age 32 (integer) 1 127.0.0.1:6379> hset user:1 sex 1 (integer) 1 127.0.0.1:6379> hset user:2 name nono (integer) 1 127.0.0.1:6379> hset user:2 age 24 (integer) 1 127.0.0.1:6379> hset user:2 sex 1 (integer) 1 127.0.0.1:6379> hset user:2 desc "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" (integer) 1 127.0.0.1:6379> hget user:1 name "aomi" 127.0.0.1:6379> hkeys user:1 1) "name" 2) "age" 3) "sex" 127.0.0.1:6379> hvals user:1 1) "aomi" 2) "32" 3) "1" 127.0.0.1:6379> hgetall user:1 1) "name" 2) "aomi" 3) "age" 4) "32" 5) "sex" 6) "1" 127.0.0.1:6379> object encoding user:1 "ziplist" 127.0.0.1:6379> object encoding user:2 "hashtable" 127.0.0.1:6379>
筆者先向redis數據庫增長倆個數據。分別爲:user:1 和user:2。同時寫了三個得到哈希信息的命令。分別爲:hget 、hkeys、hvals、hgetall。接着顯示出這個倆數據的內部編碼。
語法:
HSET key field value
從語法我就能夠看出數據結構大概是一個什麼樣子。這個JAVA的類有一個像。key爲類名,field爲成員,value:爲成員的值。事實上你能夠看到筆者上面的例子就有一點像存放User類的實例同樣子。
key:鍵名稱
field:成員名
value:成員的值
語法:
HGET key field
語法:
HKEYS key
語法:
HVALS key
語法:
HGETALL key
上面例子咱們能夠看到倆種內部編碼。user:2的desc卻很長。必定大於64個字節。什麼意思。若是成員值的長度大於64個字節的話,內部編碼都會轉爲hashtable。固然還有若是你的成員個數大於512的話,內部編碼也會轉爲hashtable有話。
筆者想知若是user:2的desc成員刪除。內部編號會不會爲ziplist。同時你們看一下什麼刪除。
127.0.0.1:6379> hdel user:2 desc (integer) 1 127.0.0.1:6379> object encoding user:2 "hashtable" 127.0.0.1:6379> hkeys user:2 1) "name" 2) "sex" 3) "age" 127.0.0.1:6379> hdel user:2 (error) ERR wrong number of arguments for 'hdel' command
最後一步出錯了。筆者就想試一下刪除整個鍵。大家看到出錯了。要刪除的話。還要用下的。
127.0.0.1:6379> del user:2 (integer) 1 127.0.0.1:6379> exists user:2 (integer) 0
這裏有一個觀念。hdel刪除的是對應哈希裏面的成員的。而要刪除key是屬於數據層的。
redis的列表有一點奇怪。在筆者第一接觸的時候。被搞得有一點暈。大家能夠這樣子理解。如今有一個列表。他只有倆個地方能夠進入。一個是左邊的頭,一個是右邊的頭。出去也只這倆個頭。
數據結果:一個列表
內部編碼:有三種:一種是ziplist,一個是linkedlist。最後一種是quicklist。
127.0.0.1:6379> lpush list1 a b c d e f (integer) 6 127.0.0.1:6379> lrange list1 0 -1 1) "f" 2) "e" 3) "d" 4) "c" 5) "b" 6) "a" 127.0.0.1:6379>
語法:
LPUSH key value [value ...]
lpush就是從左邊口進入。因此就 a進完 ,b在進,b進完了c進。以此類推。這個時候咱們要以看到列表以下
左邊進入--> f-e-d-c-b-a
語法:
LRANGE key start stop
start表示從哪裏開始,stop表示從哪裏結果,若是stop=-1表示最後一個,-2的話表示倒數第二個,-3的話表示倒數第三個,依此類推。
如今咱們就能夠明白爲何上面是f-e-d-c-b-a。咱們在來看看從右邊插入會是什麼樣子。
127.0.0.1:6379> rpush list2 1 2 3 4 5 6 7 (integer) 7 127.0.0.1:6379> lrange list2 0 -1 1) "1" 2) "2" 3) "3" 4) "4" 5) "5" 6) "6" 7) "7" 127.0.0.1:6379>
若是咱們從右邊的口進入的話,那麼在列表裏就是以下
1-2-3-4-5-6-7 <--右邊進入
因此lrange顯示就是1,2,3,4,5,6,7。
列表的內部是有序的而且能夠重複。筆者插入一個相同信的看看
127.0.0.1:6379> lpush list3 a a b c d (integer) 5 127.0.0.1:6379> lrange list3 0 -1 1) "d" 2) "c" 3) "b" 4) "a" 5) "a"
上面都是入,沒有出。如今筆者要作一出的操做。以下
127.0.0.1:6379> lrange list1 0 -1 1) "f" 2) "e" 3) "d" 4) "c" 5) "b" 6) "a" 127.0.0.1:6379> lpop list1 "f" 127.0.0.1:6379> lrange list1 0 -1 1) "e" 2) "d" 3) "c" 4) "b" 5) "a" 127.0.0.1:6379>
咱們發現用lpop的命令以後。相對應的值也會被取出。如上面的「f」就是被取出來了。筆者最後看一下列表只有:e-d-c-b-a了。原來應該是f-e-d-c-b-a。說明lpop是左邊出的。一樣子rpop是從右邊出的。
127.0.0.1:6379> rpop list1 "a" 127.0.0.1:6379> lrange list1 0 -1 1) "e" 2) "d" 3) "c" 4) "b" 127.0.0.1:6379>
接下讓咱們看一下他的內部編碼。
127.0.0.1:6379> lpush list4 a "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaadddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddd" (integer) 4 127.0.0.1:6379> object encoding list4 "quicklist" 127.0.0.1:6379> lrange list4 0 -1 1) "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaadddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddddd" 2) "a" 3) "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" 4) "a" 127.0.0.1:6379>
筆者用了很長的值,內部編碼仍是quicklist。quicklist是3.2版本以後出現的。他接結ziplist和linkedlist倆都優勢而生的。相關的大家能夠去查看官網。
redis提供列表的功能事實上是能夠當隊列和棧來用的。先進先出。那麼你就要用到lpush和rpop 或是rpush 和lpop。從左邊進,從右邊出。後進先出就是棧了。那麼就lpush和lpop或是rpush和rpop了。
他是一個無序的,同時他不能有重複。
數據結果:你能夠理解爲一個箱子。東西隨便放。但不能重複。
內部編碼:他有倆種:一是intset,一種hashtable.
例子
127.0.0.1:6379> sadd set1 a a b c d e f (integer) 6 127.0.0.1:6379> smembers set1 1) "a" 2) "d" 3) "c" 4) "f" 5) "b" 6) "e" 127.0.0.1:6379> object encoding set1 "hashtable" 127.0.0.1:6379> sadd set2 a b "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" (integer) 3 127.0.0.1:6379> object encoding set2 "hashtable" 127.0.0.1:6379> object encoding set1 "hashtable" 127.0.0.1:6379> sadd set3 1 2 3 4 5 6 7 8 9 (integer) 9 127.0.0.1:6379> smembers set3 1) "1" 2) "2" 3) "3" 4) "4" 5) "5" 6) "6" 7) "7" 8) "8" 9) "9" 127.0.0.1:6379> object encoding set3 "intset" 127.0.0.1:6379>
筆者增長了三個集合類型的數據。set1通常沒有什麼特別的。set2裏面有一個值長度很長。set3都是整數。
語法:
SADD key member [member ...]
筆者就不說明了。member就是相對應的成員值。
語法:
SMEMBERS key
筆者分別顯示出三個的內部編碼。發現只有當集合裏面所有是整數的時候,內部編碼是intset。其餘都是hashtable。
大家能夠這樣子裏面集合類型是無序的。爲了讓他有序,把集合類型的數據結構裏面加入值來表示他的順序。
數據結構:一樣子的箱子。只是在放入這個箱子的東西。必須貼上相關的數字標籤。
內部編碼:有倆種:一種是ziplist,一種是skiplist。
127.0.0.1:6379> zadd stu 80 math 60 english 70 chinese (integer) 3 127.0.0.1:6379> zcard stu (integer) 3 127.0.0.1:6379> zrange stu 0 -1 1) "english" 2) "chinese" 3) "math" 127.0.0.1:6379> zrange stu 0 -1 withscores 1) "english" 2) "60" 3) "chinese" 4) "70" 5) "math" 6) "80" 127.0.0.1:6379> zadd stu2 80 "aoooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo" (integer) 1 127.0.0.1:6379> object encoding stu "ziplist" 127.0.0.1:6379> object encoding stu2 "skiplist" 127.0.0.1:6379>
有沒有感受就是一個報表。好比學習的成績之類的數據體現。80分 數學。60分 語文。那麼前面是否是必定要是一個數字呢?筆者作了一下實驗。
127.0.0.1:6379> zadd stu3 "GODD" "AOMI" 80 EN (error) ERR value is not a valid float
看到了果真要一個數字。同時咱們能夠到他的倆種內部編碼。