每一個資料庫都有自己的數據類型。同樣子redis為我們提供了五種類型的數據——字元串、哈希、列表、集合、有序集合。我們知道關係型數據的數據存放型式是一張二維表。用行和列來表示數據之間的關係。redis是一個nosql資料庫當然不可能在用什麼二維表的形式來表示了。他所有的數據都是以key=value的 ...
每一個資料庫都有自己的數據類型。同樣子redis為我們提供了五種類型的數據——字元串、哈希、列表、集合、有序集合。我們知道關係型數據的數據存放型式是一張二維表。用行和列來表示數據之間的關係。redis是一個nosql資料庫當然不可能在用什麼二維表的形式來表示了。他所有的數據都是以key=value的形式來存放的。每一種數據類型都有數據結構和內部編碼的概念。數據結構你們可以理解他們的存放時的結構。而內部編碼就是數據結構的具體實現。但是一種數據結構可以對應多種內部編碼的實現。接下來筆者就要去看看每一種數據類型相關的操作命令和數據結構,內部編碼。
1.字元串類型的數據
字元串類型,筆者認為比較簡單的類型。
數據結果:一串值
內部編碼:一共有三種: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時面存放了三個值。他這三個值對應的鍵為:k1、k2、k3。
- set:用於增加字串符的值。
語法:
SET key value [EX seconds] [PX milliseconds] [NX|XX]
key:鍵名
value:鍵對應的值
EX:表示有效的時間。0表示永存,大於0表示在幾秒內有效。
PX:表示有效時間。只是單位是毫秒。
NX:表示不存在的時候才可以增加成功。
XX:表示只有存在的時候才可以增加成功。
- object encoing:用於查看當前鍵的內部編碼。
語法:
OBJECT subcommand [arguments [arguments]]
subcommand:這個有三個值 refcount(用於查看對象引用次數)、encoding(查看內部編碼)、idletime(存在的時間)。
- keys:用於查看當前資料庫有多少鍵。
語法:
KEYS pattern
pattern:*表示全部。*aaa表示查找以aaa結尾的。[a,b]123表示查看以a或是b,並且後面是123。相信不用筆者都說明瞭。
- get:用於獲取指定鍵的值。
語法:
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的命令是一件很簡單的事情。筆者是一邊看命令手冊一邊寫命令的。所以大家也可以這樣子。自己寫過一遍基本上都不會忘記。如果有不懂的話,查看一下手冊就可以了。至於手冊網路上很多。建義可以去官網看看。
2.哈希類型的數據
這個數據類型算是這五種類型中最為複雜的。存放形式不用說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:用於增加哈希
語法:
HSET key field value
從語法我就可以看出數據結構大概是一個什麼樣子。這個JAVA的類有一個像。key為類名,field為成員,value:為成員的值。事實上你可以看到筆者上面的例子就有一點像存放User類的實例一樣子。
key:鍵名稱
field:成員名
value:成員的值
- hget:用於獲得一個成員的值。
語法:
HGET key field
- hkeys:用於獲得當前鍵下的所有成員
語法:
HKEYS key
- hvals:用於獲得當前鍵下的所有值
語法:
HVALS key
- hgetall:用於獲得當前鍵下的所有成員和對應的所有值。
語法:
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是屬於數據層的。
3.列表類型的數據
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:將一個或多個值 從左邊插入
語法:
LPUSH key value [value ...]
lpush就是從左邊口進入。所以就 a進完 ,b在進,b進完了c進。以此類推。這個時候我們要以看到列表如下
左邊進入--> f-e-d-c-b-a
- lrange:獲得指定區間內的元素。
語法:
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了。
4.集合類型的數據
他是一個無序的,同時他不能有重覆。
數據結果:你可以理解為一個箱子。東西隨便放。但不能重覆。
內部編碼:他有倆種:一是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:用於增加一個集合。
語法:
SADD key member [member ...]
筆者就不說明瞭。member就是相對應的成員值。
- smembers用於顯示裡面的成員
語法:
SMEMBERS key
筆者分別顯示出三個的內部編碼。發現只有當集合裡面全部是整數的時候,內部編碼是intset。其他都是hashtable。
5.有序集合類型的數據
你們可以這樣子裡面集合類型是無序的。為了讓他有序,把集合類型的數據結構裡面加入值來表示他的順序。
數據結構:同樣子的箱子。只是在放入這個箱子的東西。必須貼上相關的數字標簽。
內部編碼:有倆種:一種是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
看到了果然要一個數字。同時我們可以到他的倆種內部編碼。