python-RabbitMQ基础篇

一、RabbitMQ简单介绍

  RabbitMQ是一个在AMQP基础上完整的,可复用的企业消息系统。他遵循Mozilla Public License开源协议。

MQ全称为Message Queue, 消息队列(MQ)是一种应用程序对应用程序的通信方法。应用程序通过读写出入队列的消息(针对应用程序的数据)来通信,而无需专用连接来链接它们。消 息传递指的是程序之间通过在消息中发送数据进行通信,而不是通过直接调用彼此来通信,直接调用通常是用于诸如远程过程调用的技术。排队指的是应用程序通过 队列来通信。队列的使用除去了接收和发送应用程序同时执行的要求。

  RabbitMQ安装:

  官网地址:http://www.rabbitmq.com/install-debian.html

  方法一:

echo 'deb http://www.rabbitmq.com/debian/ testing main' |
        sudo tee /etc/apt/sources.list.d/rabbitmq.list 

wget -O- https://www.rabbitmq.com/rabbitmq-release-signing-key.asc |
        sudo apt-key add - 
#Run the following command to update the package list:
#更新程序包 sudo apt-get update

   Install rabbitmq-server package:

sudo apt-get install rabbitmq-server

   启动RabbitMQ服务:

   #/etc/init.d/rabbitmq-server  

  linux安装:  

安装配置epel源
   $ rpm -ivh http://dl.fedoraproject.org/pub/epel/6/i386/epel-release-6-8.noarch.rpm
 
安装erlang
   $ yum -y install erlang
 
安装RabbitMQ
   $ yum -y install rabbitmq-server

   注意:service rabbitmq-server start/stop

  安装过程注意问题:  

安装erlang
   $ yum -y install erlang
系统报错:Error: Cannot retrieve metalink for repository: epel. Please verify its path and try again
解决方法:
打开/etc/yum.repos.d/epel.repo
找到:
1.[epel]
2.name=Extra Packages for Enterprise Linux 6 - $basearch3.#baseurl=http://download.fedoraproject.org/pub/epel/6/$basearch4.mirrorlist=https://mirrors.fedoraproject.org/metalink?repo=epel-6&arch=$basearch 修改为; 1.[epel]
2.name=Extra Packages for Enterprise Linux 6 - $basearch3.baseurl=http://download.fedoraproject.org/pub/epel/6/$basearch4.#mirrorlist=https://mirrors.fedoraproject.org/metalink?repo=epel-6&arch=$basearch
参考地址:http://xiedexu.cn/error-cannot-retrieve-metalink-repository-epel-please-verify-path-try.htm

   安装API:

pip install pika
or
easy_install pika
or
源码
 
https://pypi.python.org/pypi/pika

   使用API操作RabbitMQ

  基于Queue实现生产者消费者模型

import queue,time
import threading
#先进先出
q = queue.Queue(20)
#生产者
def productor(arg):
    """
    买票
    :param arg:
    :return:
    """
    q.put(str(arg)+"-买票")
for i in range(30):
    t = threading.Thread(target=productor,args=(i,))  #创建买票线程生产者,并将动态参数i传递给productor(arg)
    t.start()  #执行买票线程
    # if i == 30:
    #     break
#消费者
def consumer(arg):
    """
    服务器后台,
    :param arg:
    :return:
    """
    while True:
        print(arg,q.get())  #获取队列中存在元素
        time.sleep(1)
sk = threading.BoundedSemaphore(5)
for j in range(5):
    t = threading.Thread(target=consumer,args=(j,))#创建买票线程生产者,并将动态参数j传递给consumer(arg)
    t.start()  #执行c线程

   1、RabbitMQ简单操作

  对于RabbitMQ来说,生产和消费不再针对内存里的一个Queue对象,而是某台服务器上的RabbitMQ Server实现的消息队列。

   在服务器端启动RabbitMQ程序:

  生产者:

import pika

# ######################### 生产者 #########################
print('--------------------生产者------------------------')
#链接rabbit服务器
connection = pika.BlockingConnection(pika.ConnectionParameters(
        host='192.168.1.103'))
channel = connection.channel()  #创建频道

channel.queue_declare(queue='nihao001')#创建队列名为xiaoluo001
#向xiaoluo001队列插入数字,routing_key=‘xiaoluo001’为队列名,body后则是插入的数据
channel.basic_publish(exchange='',
                      routing_key='nihao001',
                      body='Hello World!123456790')
print(" [x] Sent 'Hello World!'")
#缓冲区已经flush且消息已经 确认发送到rabbitMq,最后关闭连接
connection.close() 

   消费者:

import pika
# ########################## 消费者 ##########################
print("---------------消费者----------------------")
#链接rabbitMQ
connection = pika.BlockingConnection(pika.ConnectionParameters(
        host='192.168.1.103'))
#创建链接频道
channel = connection.channel()
#创建队列名为:xiaoluo001
channel.queue_declare(queue='xiaoluo001')
#callback函数:接收服务器端发送消息,他会被pika调用
def callback(ch, method, properties, body):
    print(" [x] Received %r" % body)
#从队列中取数据,callback回调函数,如果拿到数据则执行callback函数
channel.basic_consume(callback,
                      queue='xiaoluo001',
                      no_ack=True)
print(' [*] Waiting for messages. To exit press CTRL+C')
#循环等待数据处理和callback函数处理的数据
channel.start_consuming()

   2、acknowledgment消息不丢失

  no-ack = False,如果消费者遇到情况(its channel is closed, connection is closed, or TCP connection is lost)挂掉了,那么,RabbitMQ会重新将该任务添加到队列中。

 %r用rper()方法处理对象

   %s用str()方法处理对象

  1、生产者不需要不需要变化

import pika
# ######################### 生产者 #########################
print('--------------------生产者------------------------')
#链接rabbit服务器
connection = pika.BlockingConnection(pika.ConnectionParameters(
        host='192.168.1.152'))
channel = connection.channel()  #创建频道

channel.queue_declare(queue='nihao001')#创建队列名为xiaoluo001
#向xiaoluo001队列插入数字,routing_key=‘xiaoluo001’为队列名,body后则是插入的数据
channel.basic_publish(exchange='',
                      routing_key='nihao001',
                      body='Hello World!123456790')
print(" [x] Sent 'Hello World!'")
#缓冲区已经flush且消息已经 确认发送到rabbitMq,最后关闭连接
connection.close()

    2、消费者

import pika  #导入pika模块
import time
#创建并连接rabbitmq
connection = pika.BlockingConnection(pika.ConnectionParameters(
	host="192.168.1.152"))
#创建频道
channel = connection.channel()
#如果生产者没队列,那么消费者创建队列
channel.queue_declare(queue='lcj006')
def callback(ch,method,properties,body):
	print('[xx] Receivd %r' % body)  #%r:%r用rper()方法处理对象
	import time
	time.sleep(3)
	print("ok")
	ch.basic_ack(delivery_tag = method.delivery_tag) #此代码主要表示:ch想消息对列传送数据,表示数据已经传送成功,不需再传数据

channel.basic_consume(callback,
					  queue='lcj006',
					  no_ack=False) #no_ack=False:服务器挂机,rabbitmq会将此消息放至消息队列中
print(' [*] Waiting for messages. To exit press CTRL+C')
channel.start_consuming() #取队列中取数据
#当生产者生成一条数据,别消费者接受,消费者中断后如果不超过10秒,连接池队列中的数据依然存在,
# ,否则超过10秒,消费者重新链接服务器,数据丢失,从而消费者等待

  3、durable   消息不丢失(持久化)

   durable作用:当rabbitmq挂机了,durable将队列中数据持久化,前提需告知生产者发送的数据是需要durable持久化,那么代码中就需要用到delivery_mode=2对数据进行标记

  生产者:  

import pika
#链接rabbit服务器
connection = pika.BlockingConnection(pika.ConnectionParameters(host='192.168.1.152'))
#创建频道
channel = connection.channel()
#创建队列名为hello,使用持久化durable=True
# make message persistent
channel.queue_declare(queue='hello001', durable=True) #
 #向hello队列插入数字,routing_key=‘Hello World!’,body后则是插入的数据
channel.basic_publish(exchange='',   #这个exchange参数就是这个exchange的名字. 空字符串标识默认的或者匿名的exchange:如果存在routing_key, 消息路由到routing_key指定的队列中。
                      routing_key='hello001',
                      body='Hello World!',
                      properties=pika.BasicProperties(
                          delivery_mode=2, # make message persistent
						  # 给服务器标记此条消息需要持久化,通过delivery_mode=2设置,
						 #不管mq是否挂机,此条消息都会存在服务器的消息对列中
                      ))
print(" [x] Sent '开始队列!'")
connection.close()

  消费者:

#!/usr/bin/env python
#*- coding:utf-8 -*-
import pika
print("----------------------消费者--------------------")
connection = pika.BlockingConnection(pika.ConnectionParameters(host='192.168.1.152'))
channel = connection.channel()
#创建队列名为hello,使用持久化durable=True
# make message persistent
channel.queue_declare(queue='hello001', durable=True)
#此代码主要表示:ch向消息对列传送数据,表示数据已经传送成功,不需再传数据
def callback(ch, method, properties, body):
    print(" [x] Received %r" % body)
    import time
    time.sleep(10)
    print( 'ok')
    ch.basic_ack(delivery_tag = method.delivery_tag)

channel.basic_consume(callback,
                      queue='hello001',
                      no_ack=False)
print(' [*] 等待队列. To exit press CTRL+C')
channel.start_consuming()

 注意:标记消息为持久化的并不能完全保证消息不会丢失,尽管告诉RabbitMQ保存消息到磁盘,当RabbitMQ接收到消息还没有保存的时候仍然有一个短暂的时间窗口. RabbitMQ不会对每个消息都执行同步fsync(2) --- 可能只是保存到缓存cache还没有写入到磁盘中,这个持久化保证不是很强,但这比我们简单的任务queue要好很多,如果你想很强的保证你可以使用 publisher confirms

  4、消息获取顺序

  默认消息队列里的数据是按照顺序被消费者拿走,例如:消费者1 去队列中获取 奇数 序列的任务,消费者1去队列中获取 偶数 序列的任务。

channel.basic_qos(prefetch_count=1) 表示谁来谁取,不再按照奇偶数排列

 生产者:

import pika
import sys
#链接rabbit服务器
connection = pika.BlockingConnection(pika.ConnectionParameters(host='192.168.1.152'))
#创建频道
channel = connection.channel()
#创建队列名为hello,使用持久化durable=True
# make message persistent
channel.queue_declare(queue='hello0001', durable=True) #
message=''.join(sys.argv[1:])or "nihao"
 #向hello队列插入数字,routing_key=‘Hello World!’,body后则是插入的数据
channel.basic_publish(exchange='',   #这个exchange参数就是这个exchange的名字. 空字符串标识默认的或者匿名的exchange:如果存在routing_key, 消息路由到routing_key指定的队列中。
                      routing_key='hello0001',
                      body='Hello World!',
                      properties=pika.BasicProperties(
                          delivery_mode=2, # make message persistent
						  # 给服务器标记此条消息需要持久化,通过delivery_mode=2设置,
						 #不管mq是否挂机,此条消息都会存在服务器的消息对列中
                      ))
print(" [x] Sent '开始队列!'")
connection.close()

  消费者:

#!/usr/bin/env python
#*- coding:utf-8 -*-
import pika
print("----------------------消费者--------------------")
connection = pika.BlockingConnection(pika.ConnectionParameters(host='192.168.1.152'))
channel = connection.channel()
#创建队列名为hello001,设置持久化队列durable=True
# make message persistent
channel.queue_declare(queue='hello0001', durable=True)
#此代码主要表示:ch向消息对列传送数据,表示数据已经传送成功,不需再传数据
def callback(ch, method, properties, body):
    print(" [x] Received %r" % body)
    import time
    time.sleep(10)
    print( 'ok')
    ch.basic_ack(delivery_tag = method.delivery_tag)
#表示不按照奇数偶数进行取数据,按照顺寻取数据
channel.basic_qos(prefetch_count=1)
channel.basic_consume(callback,
                      queue='hello0001',
                      no_ack=False)
print(' [*] 等待队列. To exit press CTRL+C')
channel.start_consuming()

   exchange交换

  exchange类型可用: direct , topic , headers 和 fanout 。 我们将要对最后一种进行讲解 --- fanout

   exchange type = fanout  :表示只有跟exhange绑定连接的所有队列都发消息

     exchange type = direct :队列绑定关键

   exchange type = topic  :绑定几个模糊的关键字

  5、发布订阅

  

  

发布订阅和简单的消息队列区别在于,发布订阅会将消息发送给所有的订阅者,而消息队列中的数据被消费一次便消失。所以,RabbitMQ实现发布和订阅时,会为每一个订阅者创建一个队列,而发布者发布消息时,会将消息放置在所有相关队列中。

 exchange type = fanout  

 #fanout :表示只有跟exhange绑定连接的所有队列都发消息

  1、订阅者:

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author:lcj
import pika
#连接rabbirmq服务器
connection = pika.BlockingConnection(pika.ConnectionParameters(
        host='192.168.1.152'))
#建立频道
channel = connection.channel()
#创建exchange=logs,type='fanout表示只要跟exchange绑定连接的所有对列都会收到消息
channel.exchange_declare(exchange='logs',
                         type='fanout')
#随机创建队列名,每一次创建一个队列
result = channel.queue_declare(exclusive=True)  #队列断开后自动删除临时队列
queue_name = result.method.queue  #队列名采用服务器端分配的临时队列
#绑定队列queue_name
channel.queue_bind(exchange='logs',
                   queue=queue_name)
print(' [*] Waiting for logs. To exit press CTRL+C')
def callback(ch, method, properties, body):
    print(" [x] %r" % body)
#通过callback发消息
channel.basic_consume(callback,
                      queue=queue_name,
                      no_ack=True)
channel.start_consuming()

  2、发布

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author:lcj
import pika
import sys
#简介rabbitmq
connection = pika.BlockingConnection(pika.ConnectionParameters(
        host='192.168.1.152'))
#建立频道
channel = connection.channel()
#创建exchange=logs,type='fanout表示只要跟exchange绑定连接的所有对列都会收到消息
channel.exchange_declare(exchange='logs',
                         type='fanout')
#接收参数
message = ' '.join(sys.argv[1:]) or "info: Hello World!"
#将message参数信息发到exchange里面
channel.basic_publish(exchange='logs',
                      routing_key='',
                      body=message)
print(" [x] Sent %r" % message)
connection.close()

  6、关键字发送

  

  exchange type = direct

之前事例,发送消息时明确指定某个队列并向其中发送消息,RabbitMQ还支持根据关键字发送,即:队列绑定关键字,发送者将数据根据关键字发送到消息exchange,exchange根据 关键字 判定应该将数据发送至指定队列。

  生产者:

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author:lcj
import pika
print('-----------------生产者发消息---------------')
import sys
#连接rabbitmq服务器
connection = pika.BlockingConnection(pika.ConnectionParameters(host="192.168.1.152"))
#创建频道
channel= connection.channel()
##随机创建队列名,每一次创建一个队列
channel.exchange_declare(exchange='direct_logs-test',
						type='direct')
# severity = sys.argv[1] if len(sys.argv) > 1 else 'info'
# message = ''.join(sys.argv[2:]) or 'nihao lcj'
severity = 'info'
message = '1234567890'
channel.basic_publish(exchange="direct_logs-test",
					  routing_key=severity, #关键字
					  body=message)
print(" [x] Sent %r:%r" % (severity, message))
connection.close()

  2、消费者

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author:lcj
import pika
import sys
print('---------------------消费者取消息-----------------')
#连接rabbitmq服务器
connection = pika.BlockingConnection(pika.ConnectionParameters(
        host='192.168.1.152'))
#创建频道
channel = connection.channel()
#创建exchange=logs,类型为direct
channel.exchange_declare(exchange='direct_logs-test',
                         type='direct')
#随机创建队列名,每一次创建一个队列
result = channel.queue_declare(exclusive=True)
queue_name = result.method.queue
#
# severities = sys.argv[1:]
severities = ['info',]
# if not severities:
#     sys.stderr.write("Usage: %s [info] [warning] [error]\n" % sys.argv[0])
#     sys.exit(1)
#绑定  severity为动态参数
for severity in severities:   #循环取值
    channel.queue_bind(exchange='direct_logs-test',
                       queue=queue_name,
                       routing_key=severity)  #severity传参

print(' [*] Waiting for logs. To exit press CTRL+C')
#等待取数据
def callback(ch, method, properties, body):
    print(" [x] %r:%r" % (method.routing_key, body))
#
channel.basic_consume(callback,
                      queue=queue_name,
                      no_ack=True)

channel.start_consuming()

  6、模糊匹配

  

 

   exchange type = topic

  在topic类型下,可以让队列绑定几个模糊的关键字,之后发送者将数据发送到exchange,exchange将传入”路由值“和 ”关键字“进行匹配,匹配成功,则将数据发送到指定队列。

  • # 表示可以匹配 0 个 或 多个 单词
  • *  表示只能匹配 一个 单词
发送者路由值              队列中
old.boy.python          old.*  -- 不匹配
old.boy.python          old.#  -- 匹配

  消费者:  

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author:lcj
import pika
import sys

connection = pika.BlockingConnection(pika.ConnectionParameters(
        host='localhost'))
#创建频道
channel = connection.channel()

channel.exchange_declare(exchange='topic_logs',
                         type='topic')  #创建模糊匹配

#随机创建队列
result = channel.queue_declare(exclusive=True)
queue_name = result.method.queue

binding_keys = sys.argv[1:]  #绑定关键字
if not binding_keys:  #判断关键字是否存在
    sys.stderr.write("Usage: %s [binding_key]...\n" % sys.argv[0])
    sys.exit(1)

for binding_key in binding_keys:
    channel.queue_bind(exchange='topic_logs',
                       queue=queue_name,
                       routing_key=binding_key)

print(' [*] Waiting for logs. To exit press CTRL+C')

def callback(ch, method, properties, body):
    print(" [x] %r:%r" % (method.routing_key, body))

channel.basic_consume(callback,
                      queue=queue_name,
                      no_ack=True)

channel.start_consuming()

  生产者:

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author:lcj
import pika
import sys
connection = pika.BlockingConnection(pika.ConnectionParameters(
        host='localhost'))
channel = connection.channel()

channel.exchange_declare(exchange='topic_logs',
                        type='topic')

routing_key = sys.argv[1] if len(sys.argv) > 1 else 'anonymous.info'
message = ' '.join(sys.argv[2:]) or 'Hello World!'
channel.basic_publish(exchange='topic_logs',
                      routing_key=routing_key,
                      body=message) 
print(" [x] Sent %r:%r" % (routing_key, message))
connection.close()

  

posted on 2016-07-27 00:40  lcj122  阅读(559)  评论(0编辑  收藏  举报

导航