文章目录

  • 集合(set)
  • 元素操作
  • 添加元素
  • 移除元素
  • 检查给定元素是否存在于集合
  • 返回集合的大小
  • 返回集合包含的所有元素
  • 集合的无序性质
  • 示例:赞、喜欢、Like、签到……
  • 投票功能的 API 及其实现
  • 投票功能的使用示例
  • 示例:打标签功能
  • 打标签功能的 API 及其实现
  • 打标签功能的使用示例
  • 从集合里面随机地弹出一个元素
  • 从集合里面随机地返回元素
  • SRANDMEMBER 的使用示例
  • 示例:抽奖系统
  • 抽奖程序的 API 及其实现
  • 抽奖程序的使用示例
  • 集合运算操作
  • 差集运算
  • 交集运算
  • 并集运算
  • 示例:使用集合实现共同关注功能
  • 示例:构建商品筛选功能
  • 示例:构建商品筛选功能
  • 示例:构建商品筛选功能
  • 商品筛选功能的实现及其 API
  • 商品筛选功能的使用示例
  • 复习

集合(set)

储存多个各不相同的元素。

Redis 的集合以无序的方式储存多个各不相同的元素。

用户可以快㏿地向集合添加元素,或者从集合里面 删除元素,也可以对多个集合进行集合运算操作,比如计算并集、交集和差集。

redis zset取并集_redis

元素操作

添加元素、删除元素、检查元素是否存在、返回集合的大小,等等。

添加元素

SADD key element [element …]
将一个或多个元素添加到 给定的集合里面,已经存在于集合的元素会自 动被忽略,命令返回新添加到集合的元素数量。

redis> SADD friends "peter"
(integer) 1
redis> SADD friends "jack" "tom" "john"
(integer) 3
redis> SADD friends "may" "tom"
(integer) 1

命令的复杂度为 O(N),N 为成功添加的元素数量。

redis zset取并集_redis zset取并集_02

移除元素

SREM key element [element …]
移除集合中的一个或者多个元素,不存在于集合中的元素会自 动被忽略,命令返回存在并且被移除的元素数量。
redis> SREM friends “peter”

(integer) 1
redis> SREM friends "tom" "john"
(integer) 2

命令的复杂度为 O(N),N 为被移除元素的数量。

redis zset取并集_API_03

检查给定元素是否存在于集合

SISMEMBER key element
检查给定的元素是否存在于集合,存在的 话返回 1 ;如果元素不存在,或者 给定的键不存在,那么返回 0 。

redis> SISMEMBER friends "peter"
(integer) 1
redis> SISMEMBER friends "li lei"
(integer) 0
redis> SISMEMBER NOT-EXISTS-KEY "element"
(integer) 0

命令的复杂度为 O(1) 。

redis zset取并集_redis_04

返回集合的大小

SCARD key
返回集合包含的元素数量(也即是集合的基数)。

redis> SCARD friends
(integer) 6
redis> SREM friends "peter" "jack" "tom"
(integer) 3
redis> SCARD friends
(integer) 3

因为 Redis 会储存集合的长度,所以命令的复杂度为 O(1) 。

redis zset取并集_复杂度_05

返回集合包含的所有元素

SMEMBERS key
返回集合包含的所有元素。

redis> SMEMBERS friends
1) "jack"
2) "peter"
3) "may"
4) "tom"
5) "john"
6) "ben"

命令的复杂度为 O(N) ,N 为集合的大小。

当集合的基数比较大时,执行这个命令有可能会造成服务器阻塞,将来会介绍更好的方式来迭代集合中的元素。

redis zset取并集_API_06

集合的无序性质
redis> SADD friends "peter" "jack" "tom" "john" 
"may" "ben"
(integer) 6
redis> SMEMBERS friends
1) "jack"
2) "peter"
3) "may"
4) "tom"
5) "john"
6) "ben"
redis> SADD another-friends "ben" "may" "john" 
"tom" "jack" "peter"
(integer) 6
redis> SMEMBERS another-friends
1) "may"
2) "ben"
3) "john"
4) "tom"
5) "jack"
6) "peter"

对于相同的一集元素,同一个集合命令可能会返回不同的 结果。
结论:不要使用集合来储存有序的数据。如果想要储存有序且重复的值,可以使用列表;如果想要储存有序且无重复的值,可以使用之后介绍的有序集合。

示例:赞、喜欢、Like、签到……

redis zset取并集_API_07


除了微博的赞之外,网上还有很多作用类似的功能,比如豆瓣的 “喜欢”、Facebook 的“Like”、以及一些网站上的“签到”等等。

这些功能实际上都是一种投票功能,比如 “赞”就是投一票,而“取消赞”就是撤销自己的投票。

通过使用 Redis 的集合,我们也可以实现类似的投票功能。

投票功能的 API 及其实现

redis zset取并集_redis zset取并集_08


这个投票程序的实现源码可以在 vote.py 看到。

vote.py
# encoding: utf-8

class Vote:

    def __init__(self, key, client):
        self.key = key
        self.client = client

    def cast(self, user):
        # 因为 SADD 命令会自动忽略已存在的元素
        # 所以我们无须在投票之前检查用户是否已经投票
        self.client.sadd(self.key, user)

    def undo(self, user):
        self.client.srem(self.key, user)

    def is_voted(self, user):
        if self.client.sismember(self.key, user):
            return True
        else:
            return False

    def voted_members(self):
        return self.client.smembers(self.key)
投票功能的使用示例
like = Vote('weibo::10086::like', client) # 记录 ID 为 10086 的微博的“赞”
like.cast('peter') # 'peter' 赞了这条微博
like.cast('john') # 'john' 赞了这条微博
like.cast('tom') # 'tom' 因为误操作而赞了这条微博
like.undo('tom') # 然后它通过“取消”按钮撤消了“赞”操作
like.is_voted('peter') # 返回 True ,因为 peter 已经赞过了
like.voted_members() # 返回 {'peter', 'john'} 
like.voted_count() # 返回 2
示例:打标签功能

redis zset取并集_复杂度_09


豆瓣允许你在标记一本书的同时,为 这本书添加标签,比如“Linux”、“系统 编程”、“计算机”、“C”、“编程”,等等。使用 Redis 的集合也可以实现这样的加标签功能。

打标签功能的 API 及其实现

redis zset取并集_redis zset取并集_10


这个打标签程序的实现代码可以在 tag.py 看到。

tag.py
# encoding: utf-8

class Tag:

    def __init__(self, key, client):
        self.key = key
        self.client = client

    def add(self, *tags):
        self.client.sadd(self.key, *tags)

    def remove(self, *tags):
        self.client.srem(self.key, *tags)

    def is_include(self, tag):
        return self.client.sismember(self.key, tag)

    def get_all(self):
        return self.client.smembers(self.key)

    def count(self):
        return self.client.scard(self.key)
打标签功能的使用示例
# 创建一个集合键来保存书本的标签
book_tag = Tag('Linux System Programming :: Tag', client)
# 给书本打标签
book_tag.add('linux', 'system programming', 'c', 'windows')
# 获取书本的所有标签
book_tag. get_all()
# 移除指定的标签
book_tag.remove('windows')

redis zset取并集_复杂度_11

从集合里面随机地弹出一个元素

SPOP key
随机地从集合中移除并返回一个元素,复杂度为 O(1) 。

redis> SADD friends "peter" "jack" "tom" "john" "may" "ben"
(integer) 6
redis> SPOP friends
"may"
redis> SMEMBERS friends
1) "tom"
2) "john"
3) "jack"
4) "peter"
5) "ben"
从集合里面随机地返回元素

SRANDMEMBER key [count]
如果没有给定可选的 count 参数,那么命令随机地返回集合中的一个元素。
如果给定了 count 参数,那么:
• 当 count 为正数,并且少于集合基数 时,命令返回一个包含 count 个元素的数组,数组中的每个元素各不相同。如果 count 大于或等于集合基数,那么命令返回整个集合。
• 当 count 为负数时,命令返回一个数组,数组中的元素可能会重复出现多次,而数组的长度为count 的绝对值。 和 SPOP 不同, SRANDMEMBER 不会移除被返回的元素。
命令的复杂度为 O(N),N 为被返回元素的数量。

SRANDMEMBER 的使用示例
redis> SADD friends "peter" "jack" "tom" "john" "may" "ben"
(integer) 1
redis> SRANDMEMBER friends # 随机地返回一个元素
"ben"
redis> SRANDMEMBER friends 3 # 随机地返回三个无重复的元素
1) "john"
2) "jack"
3) "peter"
redis> SRANDMEMBER friends -3 # 随机地返回三个可能有重复的元素
1) "may"
2) "peter"
3) "may"
示例:抽奖系统

redis zset取并集_redis_12


微博上的转发抽奖活动:每个参与者只需要进行转发,就有机会获得奖品。通过将参与抽奖的所有人都添加到一个集合里面,并使用SRANDMEMBER 命令来抽取获奖得主,我们也可以构建一个类似的抽奖功能。

抽奖程序的 API 及其实现

redis zset取并集_redis_13


这个抽象程序的实现代码可以在 loterry.py 找到。

lottery.py
# encoding: utf-8

class Lottery:

    def __init__(self, key, client):
        self.key = key
        self.client = client

    def add_player(self, *users):
        self.client.sadd(self.key, *users)

    def get_all_players(self):
        return self.client.smembers(self.key)

    def player_count(self):
        return self.client.scard(self.key)

    def draw(self, n):
        return self.client.srandmember(self.key, n)
抽奖程序的使用示例
# 创建一个赠送 Redis 书的抽奖活动
book_loterry('loterry::redis_book', client) 
# 不断地添加参与者
book_loterry.add_player('peter', 'jack', 'tom', 'may', ..., 'ben')
# 抽出三名幸运的获奖者
book_loterry.draw(3)

集合运算操作

计算并集、交集和差集。

差集运算

redis zset取并集_redis_14

交集运算

redis zset取并集_redis_15

并集运算

redis zset取并集_redis_16

示例:使用集合实现共同关注功能

redis zset取并集_redis zset取并集_17


新浪微博的共同关注功能:当用 户访问另一个用户时,该功能会显示出两个用户关注了哪些相同的用户。如果我们把每个用户的关注对象都分别放到一个集合里面的话,那么我们可以使用 SINTER 或者 SINTERSTORE 命令来实现这个共同关注功能。

例如:

peter = {‘john’, ‘jack’, ‘may’}

ben = {‘john’, ‘jack’, ‘tom’}

那么 peter 和 ben 的共同关注为:

SINTER peter ben = {‘john’, ‘jack’}

还可以使用 SINTERSTORE 来避免重复计算。

示例:构建商品筛选功能

redis zset取并集_复杂度_18


京东上的一个手机筛选系统,用户可以根据品牌、价格、网 络、屏幕尺寸等条件,对商品进 行筛选。 对于每个条件的每个选项,我们可以使用一个集合来保存 该选项对应的所有商品,并通过在多个选项之间进行交集计算,从而达到筛选的目的。

示例:构建商品筛选功能

对于品牌条件,我们可以创建这样的集合,比如 apple = { ‘iPhone 5c’, ‘iPhons 5s’, …} ,而 samsung = {‘Galaxy S5’, ‘Galaxy S4’, …} ,诸如此类。 而对于价格条件,我们可以创建这样的集合,比如 price_2900_to_4099 = { ‘Galaxy S5’, ‘Galaxy S4’, …} ,以及 price_uppon_4100 = { ‘iPhone 5c’, ‘iPhone 5S’, …} ,诸如此类。
每当用户添加一个过滤选项时,我们就计算出所有被选中选项的交集,而交集的计算结果就是符合筛 选条件的商品。

示例:构建商品筛选功能

redis zset取并集_redis_19


ZINTER apple price_uppon_5200 unicom_wcdma‘iPhone 5s 32G’当然,每次过滤都要计算一次交集的话,㏿度就太慢了,因此我们可以预先计算好,然后把结果储存在一个固定的地方,比如:

ZINTERSTORE
 apple&price_uppon_5200&unicom_wcdma
 apple price_uppon_5200 unicom_wcdma
 SMEMBER
 apple&price_uppon_5200&unicom_wcdma
商品筛选功能的实现及其 API

redis zset取并集_redis zset取并集_20


这个商品筛选功能的实现代码可以在 item_filter.py 看到。

#item_filter.py
# encoding: utf-8

class ItemFilter:

    def __init__(self, client):
        self.client = client
        self.options = set()

    def add_option(self, item_set):
        self.options.add(item_set)

    def result(self):
        return self.client.sinter(*self.options)

    def store_result(self, key):
        return self.client.sinterstore(key, *self.options)
商品筛选功能的使用示例
# 创建一个筛选器
filter = ItemFilter(client)
# 只保留 Apple 公司的产品
filter.add_option(apple_product_set)
filter.result()
# 只保留 Apple 公司生产,并且售价 5200 块以上的商品
filter.add_option(price_uppon_5200_product_set)
filter.result()
# 储存起 Apple 公司生产的、售价 5200 块以上并且支持联通 WCDMA 网络的商品
filter.add_option(unicom_wcdma_product_set)
filter.store_result('apple&price_uppon_5200&unicom_wcdma')

复习

集合可以以无序的方式 储存多个各不相同的元素,并且可以 执行添加元素、删除元素、获取所有元素等操作,以及交集、并集、差集等集合运算。

redis zset取并集_复杂度_21