您现在的位置是:首页 > 文章详情

Redis 多方式实现计数器功能

日期:2019-10-16点击:389

【大咖・来了 第7期】10月24日晚8点观看《智能导购对话机器人实践》

计数器在很多网站中都进行了广泛的应用,比如文章的点赞数、页面的浏览数、网站的访客数、视频的播放数等等。在这篇文章里,我会使用 Redis 的三种数据类型,来分别实现计数器的功能。

请跟随我一起来看看吧。

使用字符串键

下面代码演示了如何利用 Redis 中的字符串键来实现计数器功能。其中,incr() 方法用于累加计数,get_cnt() 方法用于获取当前的计数值。

 
  1. from redis import Redis 
  2.  
  3. class Counter: 
  4.     def __init__(self, client: Redis, key: str): 
  5.         self.client = client 
  6.         self.key = key 
  7.  
  8.     def incr(self, amount=1): 
  9.         """计数累加""" 
  10.         self.client.incr(self.key, amount=amount) 
  11.  
  12.     def decr(self, amount=1): 
  13.         """计数累减""" 
  14.         self.client.decr(self.key, amount=amount) 
  15.  
  16.     def get_cnt(self): 
  17.         """获取当前计数的值""" 
  18.         return self.client.get(self.key
  19.  
  20.  
  21. if __name__ == '__main__'
  22.     client = Redis(decode_responses=True
  23.     counter = Counter(client, 'page_view:12'
  24.     counter.incr() 
  25.     counter.incr() 
  26.     print(counter.get_cnt())  # 2 

假设我们要统计 page_id 为 12 的页面的浏览数,那么我们可以设定 key 为 page_view:12,用户每一次浏览,就调用一次 counter 的 incr() 方法进行计数。

使用哈希键

在上面的代码中,我们需要针对每个统计项,都单独设置一个字符串键。那么,下面我们来看看如何通过 Redis 的哈希键,来对关联的统计项进行统一管理。

 
  1. from redis import Redis 
  2.  
  3. class Counter: 
  4.     def __init__(self, client: Redis, key: str, counter: str): 
  5.         self.client = client 
  6.         self.key = key 
  7.         self.counter = counter 
  8.  
  9.     def incr(self, amount=1): 
  10.         """计数累加""" 
  11.         self.client.hincrby(self.key, self.counter, amount=amount) 
  12.  
  13.     def decr(self, amount=1): 
  14.         """计数累减""" 
  15.         self.client.hincrby(self.key, self.counter, amount=-amount) 
  16.  
  17.     def get_cnt(self): 
  18.         """获取当前计数的值""" 
  19.         return self.client.hget(self.key, self.counter) 
  20.  
  21.  
  22. if __name__ == '__main__'
  23.     client = Redis(decode_responses=True
  24.     counter = Counter(client, 'page_view''66'
  25.     counter.incr() 
  26.     counter.incr() 
  27.     print(counter.get_cnt())  # 2 

如果采用哈希键,那么,我们对于同一类型的计数,可以使用一个相同的 key 来进行存储。比如,在上面例子中,我们使用 page_view 来统计页面的浏览数,对于 page_id 为 66 的页面,直接添加到 page_view 对应的字段中即可。

使用集合键

在上面两个例子中,当动作被执行时,程序可以调用一次 incr() 累加计数的方法。某些场景下,我们可能需要对特定的动作,仅仅计数一次。什么叫“仅仅计数一次”?就是说,同一个用户/IP,多次访问某个页面,计数器只会将计数值增加 1。来看看以下代码:

 
  1. from redis import Redis 
  2.  
  3. class Counter: 
  4.     def __init__(self, client: Redis, key: str): 
  5.         self.client = client 
  6.         self.key = key 
  7.  
  8.     def add(self, item: str) -> bool: 
  9.         """计数累加,若计数之前item已存在,放回False;否则返回True""" 
  10.         return self.client.sadd(self.key, item) == 1 
  11.  
  12.     def get_cnt(self): 
  13.         """获取当前计数的值""" 
  14.         return self.client.scard(self.key
  15.  
  16.  
  17. if __name__ == '__main__'
  18.     client = Redis(decode_responses=True
  19.     counter = Counter(client, 'uv'
  20.     counter.add('user1'
  21.     counter.add('user2'
  22.     counter.add('user1')  # 重复放入 
  23.     print(counter.get_cnt())  # 2 

在实际应用中,以上代码需要稍作改动,但基本的思路不变。怎么样,你学会了吗?

原文链接:http://database.51cto.com/art/201910/604369.htm
关注公众号

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。

持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。

转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。

文章评论

共有0条评论来说两句吧...

文章二维码

扫描即可查看该文章

点击排行

推荐阅读

最新文章