集合類型介紹
集合類型也是體現redis一個比較高價值的一個類型了。因為Redis的集合類型,所以我們可以很容易的在Redis中執行差集運算、交集運算、并集運算。
首先我們先介紹一下集合類型和列表類型的區別,其實學過面向對象的語言的同學應該都能猜到這些類型有什么不同。
①集合類型和列表類型還是都能存儲2^32-1個字符串
②集合類型是無序的,列表類型是有序的
③集合類型是唯一的,列表類型的值是不唯一的
下面我們一起來看一下語法。
1、增加刪除元素命令
1
2
|
sadd key member [member ...] srem key member [member ...] |
sadd
命令是用來向集合中增加元素的,當然根據前面的學習經驗,如果集合不存在,肯定是會自動創建的。但是這里有點需要注意的是,如果元素已存在,那么他會忽略而不會覆蓋。返回值是成功加入元素的數量(忽略的元素不算)。
srem
命令也是一樣,刪除元素,如果元素存在才會刪除成功,返回值是成功刪除的元素的個數。
2、獲得集合中的所有元素
1
|
smembers key |
3、判斷元素是否在集合中
1
|
sismember key member |
這個判斷操作的時間復雜度是O(1),無論集合中的元素有多少,這個命令始終都很很快的返回結果。存在返回1,不存在或沒有這個鍵的時候都返回0 。
4、集合間運算
1
2
3
|
sdiff key [ key ...] sinter key [ key ...] sunion key [ key ...] |
這三個命令才能使Redis大顯身手!
我們先準備點測試數據。
1
2
3
4
5
6
|
127.0.0.1:6379> sadd setA 1 2 3 ( integer ) 3 127.0.0.1:6379> sadd setB 2 3 4 ( integer ) 3 127.0.0.1:6379> sadd setC 3 4 5 ( integer ) 3 |
1
2
3
4
5
|
127.0.0.1:6379> smembers setA 1) "1" 2) "2" 3) "3" 127.0.0.1:6379> |
(1)下面我們來介紹sdiff
命令,這個命令實際上是差集運算。
集合A與集合B的差值運算表示為A-B,代表所有屬于A且不屬于B的元素構成的集合。這個命令同時支持傳入多個鍵,意思是A與B先做差集運算,然后結果再與C做差集運算。
1
2
3
4
5
6
|
127.0.0.1:6379> sdiff setA setB 1) "1" 127.0.0.1:6379> sdiff setB setC 1) "2" 127.0.0.1:6379> sdiff setA setB setC 1) "1" |
(2)接下來我們介紹sinter
命令,這個命令用來對多個集合執行交集運算。集合A與集合B的交集表示為A∩B,即所有屬于A且屬于B的元素構成的集合。這個命令也同時支持傳入多個鍵,也是一樣,一步一步的做交集運算。
1
2
3
4
5
|
127.0.0.1:6379> sinter setA setB 1) "2" 2) "3" 127.0.0.1:6379> sinter setA setB setC 1) "3" |
(3)最后我們介紹并集運算命令sunion
。集合A與集合B的并集意思是既屬于A也屬于B的集合。同樣支持多個鍵傳入。
1
2
3
4
5
6
7
8
9
10
11
|
127.0.0.1:6379> sunion setA setB 1) "1" 2) "2" 3) "3" 4) "4" 127.0.0.1:6379> sunion setA setB setC 1) "1" 2) "2" 3) "3" 4) "4" 5) "5" |
5、獲得集合中元素的個數
1
2
3
4
5
|
scard key 127.0.0.1:6379> scard setA ( integer ) 3 127.0.0.1:6379> scard setB ( integer ) 3 |
這條命令用來獲取集合中元素的個數。同樣集合不存在返回0 。
6、進行集合運算并將結果存儲
1
2
3
|
sdiffstore destination key [ key ...] sinterstore destination key [ key ...] sunionstore destination key [ key ...] |
這里在所有的集合命令之后加了個store
,很清楚就是存儲的意思了。destination
就是目的地,也就是我們要存儲的鍵名。
例:
1
2
3
4
5
6
7
8
|
127.0.0.1:6379> sunionstore setAll setA setB setC ( integer ) 5 127.0.0.1:6379> smembers setAll 1) "1" 2) "2" 3) "3" 4) "4" 5) "5" |
7、隨機獲得集合中的元素
1
|
srandmember key [ count ] |
這個命令用來隨機從集合中獲取一個元素,count參數用來一詞獲取多個元素,根據count的正負有不同的意義。
(1)當count為正數時,獲取count個不重復的元素,如果count大于所有值的個數,則返回全部元素。
(2)負值時,獲得|count|個元素,但有可能相同。
其實這個隨機并不是非常隨機,因為實際上redis的集合用的是散列的存儲結構,有興趣的讀者可以自己深入研究一下。
1
2
3
4
5
6
7
8
9
10
11
12
|
127.0.0.1:6379> srandmember setA 1 1) "1" 127.0.0.1:6379> srandmember setA 2 1) "3" 2) "2" 127.0.0.1:6379> SRANDMEMBER setA 5 1) "1" 2) "2" 3) "3" 127.0.0.1:6379> SRANDMEMBER setA -2 1) "3" 2) "3" |
8、從集合中彈出一個元素
1
|
spop key |
之前我們也看過lpop與rpop,那是從列表中彈出,但是集合是無序的,所以spop是隨機彈出。 (彈出的元素被刪除,不在原來的集合中)
1
2
3
4
5
|
127.0.0.1:6379> spop setA 1 1) "2" 127.0.0.1:6379> smembers setA 1) "1" 2) "3" |
下面我們來舉例一個集合的具體應用。
在我們文章存儲的時候有個標簽tag,這個我們在關系數據庫的時候都需要使用中間表關聯。但是在Redis中,我們可以很方便的處理。
對每篇文章,我們使用一個post:文章ID:tags
的鍵,類型為集合,存儲一篇文章所屬的tags,這樣本身關系型數據庫我們需要關聯文章表、標簽表、中間關系表三個表的復雜操作,在Redis中就很好處理了。
有時候我們還需要獲取指定標簽的所有文章,我們還需要增加一類的鍵,即tag:標簽名稱:posts
的集合類型鍵,存儲每個標簽的文章ID集合,這樣我們每次想要獲取所有屬于某個標簽的文章列表的時候也不用像關系數據庫那樣關聯那么多表,而是直接通過鍵就可以直接獲得了。
而在我們想要獲取比如同時屬于java和redis標簽的文章時,我們只需要對tag:java:posts
和tag:redis:posts
做交集運算就OK了,是不是很方便呢?
總結
以上就是這篇文章的全部內容了,希望能對大家的學習或者工作帶來一定的幫助,如果有疑問大家可以留言交流。