0

0

python下使用redis构造一个简单的队列(翻译)

php中文网

php中文网

发布时间:2016-06-07 16:41:31

|

1873人浏览过

|

来源于php中文网

原创

本文展示了如何使用redis构建一个简单的多生产者,多消费者队列并且提供类似python标准库queue一样的接口。你可以使用这个队列方便的从多个进程或者耗时的计算到多个消费者进程之间共享数据。 我们使用redis列表来保存数据。redis列表按照字符串插入的顺序保

本文展示了如何使用redis构建一个简单的多生产者,多消费者队列并且提供类似python标准库queue一样的接口。你可以使用这个队列方便的从多个进程或者耗时的计算到多个消费者进程之间共享数据。

我们使用redis列表来保存数据。redis列表按照字符串插入的顺序保存数据。

下面的redis命令会被用到:

  • rpush 在列表的末尾插入一个元素
  • blpop 从列表开头获取一个元素,如果列表是空则阻塞
  • lpop 从列表开头获取一个元素,如果列表是空则返回空
  • llen 返回列表的长度

实现过程使用了redis-py库和服务器进行交互:

RedisQueue.py

立即学习Python免费学习笔记(深入)”;

import redis
class RedisQueue(object):
    """Simple Queue with Redis Backend"""
    def __init__(self, name, namespace='queue', **redis_kwargs):
        """The default connection parameters are: host='localhost', port=6379, db=0"""
        self.__db= redis.Redis(**redis_kwargs)
        self.key = '%s:%s' %(namespace, name)
    def qsize(self):
        """Return the approximate size of the queue."""
        return self.__db.llen(self.key)
    def empty(self):
        """Return True if the queue is empty, False otherwise."""
        return self.qsize() == 0
    def put(self, item):
        """Put item into the queue."""
        self.__db.rpush(self.key, item)
    def get(self, block=True, timeout=None):
        """Remove and return an item from the queue.
        If optional args block is true and timeout is None (the default), block
        if necessary until an item is available."""
        if block:
            item = self.__db.blpop(self.key, timeout=timeout)
        else:
            item = self.__db.lpop(self.key)
        if item:
            item = item[1]
        return item
    def get_nowait(self):
        """Equivalent to get(False)."""
        return self.get(False)

使用:

>>> from RedisQueue import RedisQueue
>>> q = RedisQueue('test')
>>> q.put('hello world')

现在我们使用redis-cli客户端查看数据库,期望的结果如下:

redis 127.0.0.1:6379> keys *
1) "queue:test"
redis 127.0.0.1:6379> type queue:test
list
redis 127.0.0.1:6379> llen queue:test
(integer) 1
redis 127.0.0.1:6379> lrange queue:test 0 1
1) "hello world"

我们可以使用一个不同的脚本来获取数据:

>>> from RedisQueue import RedisQueue
>>> q = RedisQueue('test')
>>> q.get()
'hello world'

随后的q.get()调用会一直阻塞直到某人重新向队列发送一个新的数据。

接下来的工作将是到队列的 编码/解码(例如python-json),这样你就可以不受限制的发送任何字符串。

Replit Ghostwrite
Replit Ghostwrite

一种基于 ML 的工具,可提供代码完成、生成、转换和编辑器内搜索功能。

下载

现在已经存在漂亮而又简单的hotqueue库,它具有像上面例子中的接口别且提供编码/解码功能。

其他值得提到的使用redis做后端的有:

  • flask-redis flask里使用redis做后端的一个基本的消息队列。
  • celery 一个基于分布式消息传递的异步任务队列/工作队列。比其他类库更高级点,可以配合不同的后端工作。
  • rq 简单的python类库作用是队列化任务并且在后端使用消费者进程处理它们。
  • resque 一个使用redis做后端的ruby库,主要为了创建后台工作,把他们放到多个队列,稍后处理他们。github在使用,并且有一个漂亮的web监控页面。
  • pyres python下resque的克隆版。

原文中的一个小bug,如下:

生产者:

In [1]: from RedisQueue import RedisQueue
In [2]: q = RedisQueue('test')
In [3]: q.put('1')
In [4]: q.put('2')

消费者:

In [10]: from RedisQueue import RedisQueue
In [11]: q = RedisQueue('test')
In [12]: q.get()
Out[12]: '1'
In [13]: q.get_nowait()
IndexError                                Traceback (most recent call last)
 in ()
----> 1 q.get_nowait()
/data9/RedisQueue.py in get_nowait(self)
     36     def get_nowait(self):
     37         """Equivalent to get(False)."""
---> 38         return self.get(False)
     39
/data9/RedisQueue.py in get(self, block, timeout)
     31
     32         if item:
---> 33             item = item[1]
     34         return item
     35
IndexError: string index out of range

使用redis-py库连接redis数据库,获取下数据,很容易就看到问题出在哪里。

In [55]: r = redis.Redis()
In [56]: r.blpop('queue:test')
Out[56]: ('queue:test', '1')
In [57]: r.lpop('queue:test')
Out[57]: '2'

当使用非阻塞方式获取数据时,redis客户端返回的是一个string;当使用阻塞方式获取数据时,redis返回的数据是一个tuple。因此,修改下get函数中对item的判断条件就可以了:

if isinstance(item,tuple):
            item = item[1]

自言自语:

翻译文章是个劳心的活,记得2、3年前有一次想翻译一篇关于nginx的文章,当时是越翻译越别扭,感觉整个人都不好了,后来就断了翻译文档的心。就在翻译这篇文章之前我还在纠结到底译不译,看当我咬咬牙,不也是顺利完活了。学技术贵在坚持,搞IT只要不是底层内核、算法层面的东西,多练练都是没问题的。

原文地址:http://peter-hoffmann.com/2012/python-simple-queue-redis-queue.html

相关文章

python速学教程(入门到精通)
python速学教程(入门到精通)

python怎么学习?python怎么入门?python在哪学?python怎么学才快?不用担心,这里为大家提供了python速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

相关标签:

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
菜鸟裹裹入口以及教程汇总
菜鸟裹裹入口以及教程汇总

本专题整合了菜鸟裹裹入口地址及教程分享,阅读专题下面的文章了解更多详细内容。

0

2026.01.22

Golang 性能分析与pprof调优实战
Golang 性能分析与pprof调优实战

本专题系统讲解 Golang 应用的性能分析与调优方法,重点覆盖 pprof 的使用方式,包括 CPU、内存、阻塞与 goroutine 分析,火焰图解读,常见性能瓶颈定位思路,以及在真实项目中进行针对性优化的实践技巧。通过案例讲解,帮助开发者掌握 用数据驱动的方式持续提升 Go 程序性能与稳定性。

9

2026.01.22

html编辑相关教程合集
html编辑相关教程合集

本专题整合了html编辑相关教程合集,阅读专题下面的文章了解更多详细内容。

56

2026.01.21

三角洲入口地址合集
三角洲入口地址合集

本专题整合了三角洲入口地址合集,阅读专题下面的文章了解更多详细内容。

51

2026.01.21

AO3中文版入口地址大全
AO3中文版入口地址大全

本专题整合了AO3中文版入口地址大全,阅读专题下面的的文章了解更多详细内容。

397

2026.01.21

妖精漫画入口地址合集
妖精漫画入口地址合集

本专题整合了妖精漫画入口地址合集,阅读专题下面的文章了解更多详细内容。

118

2026.01.21

java版本选择建议
java版本选择建议

本专题整合了java版本相关合集,阅读专题下面的文章了解更多详细内容。

3

2026.01.21

Java编译相关教程合集
Java编译相关教程合集

本专题整合了Java编译相关教程,阅读专题下面的文章了解更多详细内容。

16

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

11

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 14万人学习

Django 教程
Django 教程

共28课时 | 3.4万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号