之前安裝好了Redis之后,對于如何使用,還不是特別清楚,因此百度了下,同時結合了下自己常用的命令,做了下整合,
**以下是一些相關常用的命令;**
# 1??Redis數據類型及應用場景
Redis最為常用的數據類型主要有以下五種:
·????????String
·????????Hash
·????????List
·????????Set
·????????Sortedset
- 下圖為Redis內部內存管理中是如何描述這些不同數據類型的:

- 首先Redis內部使用一個redisObject對象來表示所有的key和value,redisObject最主要的信息如上圖所示:type代表一個value對象具體是何種數據類型,encoding是不同數據類型在redis內部的存儲方式,比如:type=string代表value存儲的是一個普通字符串,那么對應的encoding可以是raw或者是int,如果是int則代表實際redis內部是按數值型類存儲和表示這個字符串的,當然前提是這個字符串本身可以用數值表示,比如:"123" "456"這樣的字符串。
- 這里需要特殊說明一下vm字段,只有打開了Redis的虛擬內存功能,此字段才會真正的分配內存,該功能默認是關閉狀態的,該功能會在后面具體描述。通過上圖我們可以發現Redis使用redisObject來表示所有的key/value數據是比較浪費內存的,當然這些內存管理成本的付出主要也是為了給Redis不同數據類型提供一個統一的管理接口,實際作者也提供了多種方法幫助我們盡量節省內存使用,我們隨后會具體討論。
### 1.1String類型
**常用命令:**
set,get,decr,incr,mget等。
**應用場景:**
String是最常用的一種數據類型,普通的key/value存儲都可以歸為此類,這里就不所做解釋了。
**實現方式:**
String在redis內部存儲默認就是一個字符串,被redisObject所引用,當遇到incr,decr等操作時會轉成數值型進行計算,此時redisObject的encoding字段為int。
### 1.2List類型
**常用命令:**
lpush,lpop,rpop,lrange等。
**應用場景:**
Redislist的應用場景非常多,也是Redis最重要的數據結構之一,比如twitter的關注列表,粉絲列表等都可以用Redis的list結構來實現,比較好理解,這里不再重復。
**實現方式:**
Redislist的實現為一個雙向鏈表,即可以支持反向查找和遍歷,更方便操作,不過帶來了部分額外的內存開銷,Redis內部的很多實現,包括發送緩沖隊列等也都是用的這個數據結構。
### 1.3Set類型
**常用命令:**
sadd,spop,smembers,sunion等。
**應用場景:**
Redisset對外提供的功能與list類似是一個列表的功能,特殊之處在于set是可以自動排重的,當你需要存儲一個列表數據,又不希望出現重復數據時,set是一個很好的選擇,并且set提供了判斷某個成員是否在一個set集合內的重要接口,這個也是list所不能提供的。
**實現方式:**
set 的內部實現是一個 value永遠為null的HashMap,實際就是通過計算hash的方式來快速排重的,這也是set能提供判斷一個成員是否在集合內的原因。
### 1.4Sorted Set類型
**常用命令:**
zadd,zrange,zrem,zcard等
**使用場景:**
Redissorted set的使用場景與set類似,區別是set不是自動有序的,而sorted set可以通過用戶額外提供一個優先級(score)的參數來為成員排序,并且是插入有序的,即自動排序。當你需要一個有序的并且不重復的集合列表,那么可以選擇sorted set數據結構,比如twitter 的public timeline可以以發表時間作為score來存儲,這樣獲取時就是自動按時間排好序的。
**實現方式:**
Redissorted set的內部使用HashMap和跳躍表(SkipList)來保證數據的存儲和有序,HashMap里放的是成員到score的映射,而跳躍表里存放的是所有的成員,排序依據是HashMap里存的score,使用跳躍表的結構可以獲得比較高的查找效率,并且在實現上比較簡單。
### 1.5Hash類型
**常用命令:**
hget,hset,hgetall等。
**應用場景:**
我們簡單舉個實例來描述下Hash的應用場景,比如我們要存儲一個用戶信息對象數據,包含以下信息:
用戶ID為查找的key,存儲的value用戶對象包含姓名,年齡,生日等信息,如果用普通的key/value結構來存儲,主要有以下2種存儲方式:

第一種方式將用戶ID作為查找key,把其他信息封裝成一個對象以序列化的方式存儲,這種方式的缺點是,增加了序列化/反序列化的開銷,并且在需要修改其中一項信息時,需要把整個對象取回,并且修改操作需要對并發進行保護,引入CAS等復雜問題。

第二種方法是這個用戶信息對象有多少成員就存成多少個key-value對兒,用用戶ID+對應屬性的名稱作為唯一標識來取得對應屬性的值,雖然省去了序列化開銷和并發問題,但是用戶ID為重復存儲,如果存在大量這樣的數據,內存浪費還是非常可觀的。
那么Redis提供的Hash很好的解決了這個問題,Redis的Hash實際是內部存儲的Value為一個HashMap,并提供了直接存取這個Map成員的接口,如下圖:

也就是說,Key仍然是用戶ID, value是一個Map,這個Map的key是成員的屬性名,value是屬性值,這樣對數據的修改和存取都可以直接通過其內部Map的Key(Redis里稱內部Map的key為field), 也就是通過 key(用戶ID) + field(屬性標簽) 就可以操作對應屬性數據了,既不需要重復存儲數據,也不會帶來序列化和并發修改控制的問題。很好的解決了問題。
這里同時需要注意,Redis提供了接口(hgetall)可以直接取到全部的屬性數據,但是如果內部Map的成員很多,那么涉及到遍歷整個內部Map的操作,由于Redis單線程模型的緣故,這個遍歷操作可能會比較耗時,而另其它客戶端的請求完全不響應,這點需要格外注意。
**實現方式:**
上面已經說到Redis Hash對應Value內部實際就是一個HashMap,實際這里會有2種不同實現,這個Hash的成員比較少時Redis為了節省內存會采用類似一維數組的方式來緊湊存儲,而不會采用真正的HashMap結構,對應的value redisObject的encoding為zipmap,當成員數量增大時會自動轉成真正的HashMap,此時encoding為ht。
# 2??Redis數據類型相關命令
### 2.1String命令
字符串數據類型 string
字符串類型是redis基本數據類型,能存儲任何形式的字符串,包括二進制數據。
set key value 賦值
get key 取值
incr key 遞增數字(所有redis命令都是原子操作)
redis鍵命名實踐 “對象類型:對象id:對象屬性”,對于多個單詞推薦用.分割。如鍵user:1:friends表示id為1的用戶的好友列表。
incrby key increment 增加指定整數
decrby key decrement 減少指定的整數
incrbyfloat key increment 增加指定浮點數
append key value 向尾部追加值,返回追加后字符串的長度
strlen key 返回鍵值的長度
mget key [key ...] 獲取多個健值
mset key value [key value ...] 設置多個鍵值
位操作
getbit key offset 獲得一個字符串指定位置的二進制位的值(0或1)
setbit key offset value 設置字符串類型鍵指定位置的二進制值,返回該位置的舊值
bitcount key [start] [end] 統計字符串類型中值為1的二進制位個數,可以指定字節的范圍
bitop operation destkey key [key...] (and, or, xor, not)
### 2.2List命令
列表類型可以存儲一個有序的字符串列表,常用的操作是向列表兩端添加元素或者獲得某個列表的某一個片段。
內部使用雙向鏈表實現,獲取越接近兩端的元素速度就越快,不過索引訪問元素比較慢。列表類型能非常快速地完成關系數據庫難以應付的場景,如社交網絡新鮮事。
lpush key value [value...] 從列表左邊增加元素
rpush key value [value...] 從列表右邊增加元素
lpop key 從列表左邊彈出元素
rpop key 從列表右邊彈出元素
llen key 獲取列表中元素的個數
lrange key start stop 獲取列表片段(包括stop,支持負數表示從最右邊開始計數)
lrem key count value 刪除前count個值為value的元素(count>0時從左邊開始刪除,count<0從右邊開始刪除,count=0刪除所有為value的元素)
lindex key index 獲取指定索引的元素值
lset key index value 設置指定索引的元素值
ltrim key start end 只保留指定片段
linsert key fefore|after pivot value 首先從左到右查找pivot元素,再根據第二個參賽將value插入該元素前面或后面。
rpoplpush source destination 將一個元素轉移到另一個列表 ,原子操作。當source和destination相同時會不斷將對尾元素移到隊首,實現網站監控系統。
### 2.3Set命令
集合類型每個元素不同,且無序。
sadd key member [member...] 增加元素,返回成功加入元素的個數
srem key member [member...] 刪除元素
smembers key 獲得集合中所有元素
sismember key member 判斷是否存在集合中
集合間運算
sdiff [destination] key [key ...] 多個集合求差運算a-b,并存儲到destination中
sinter [destination] key [key ...] 多個集合執行交運算
sunion [destination] key [key...] 多個集合求并運算
scard key 集合中元素個數
srandmemeber key [count] 隨機獲得集合中元素(當count>0時隨機獲取count個不重復元素,count<0時不保證重復)
spop key 從集合中隨機彈出一個元素
### 2.4Sorted Set命令
列表類型通過鏈表實現,獲取靠近兩端的數據速度極快,當元素增加后中間元素比較慢,更適合"新鮮事"或“日志”這樣很少訪問中間元素的應用。
有序集合類型通過散列表和跳躍表實現的,所以即使讀取位于中間位置也很快o(nlgn)。
列表中不能簡單調整某個元素位置,有序集合可以。有序集合更耗費內存。
zadd key score member [score member...] 加入一個元素和該元素的分數(分數可以是整數或小數,+inf和-inf表示正負無窮)
zscore key member 獲得元素的分數
zrange key start stop [withscore] 按照從從小到大的順序返回start和stop之間所有元素(withscore表示帶上分數)復雜度o(logn+m)
zrevrange key start stop [withscore] 從大到小的順序
zrangebyscore key min max [withscore] [limit offset count] 按照從小到大返回分數在min和max之間的元素
zincrby key increment member 增加某個元素的分數
zcard key 獲得集合中元素個數
zcount key min max 獲得指定范圍內的元素個數
zrem key member [member ...] 刪除一個或多個元素
zremrangebyrank key start stop 按照元素分數從小到大的順序刪除指定排名范圍內的所有元素,并返回刪除元素的個數
zremrangebyscore key min max 刪除指定分數范圍內的所有元素
zrank key member 獲得元素的排名
zrevrank key member
### 2.5Hash命令
散列類型的鍵值也是一種字典結構,其存儲了字段和字段值的映射,但字段值只能是字符串,不支持其他類型。(集合類型也不支持數據類型嵌套)
hset key field value 賦值(插入時返回1,更新時返回0)
hget key field 取值
hmset key field value [field value ...]
hmget key field [feild...]
hgetall key
hexists key field 判斷字段是否存在
hsetnx key field value 當字段不存在時賦值
hincrby key field increment 增加數字
hdel key field [field...] 刪除字段
hkeys key 只獲取字段名
hvals key 只獲取字段值
hlen key 獲得字段數量