一、认识MQ
MQ全称为Message Queue 消息队列(MQ)是一种应用程序对应用程序的通信方法。MQ是消费-生产者模型的一个典型的代表,一端往消息队列中不断写入消息,而另一端则可以读取队列中的消息。这样发布者和使用者都不用知道对方的存在。
生产者消费者模式是通过一个容器来解决生产者和消费者的强耦合问题。生产者和消费者彼此之间不直接通讯,而通过阻塞队列来进行通讯,所以生产者生产完数据之后不用等待消费者处理,直接扔给阻塞队列,消费者不找生产者要数据,而是直接从阻塞队列里取,阻塞队列就相当于一个缓冲区,平衡了生产者和消费者的处理能力。
如下图所示:
二、为什么要MQ
消息队列中间件是分布式系统中重要的组件,主要解决应用解耦,异步消息,流量削锋等问题,实现高性能,高可用,可伸缩和最终一致性架构。目前使用较多的消息队列有ActiveMQ,RabbitMQ,Kafka,RocketMQ。
接下来利用一个外卖系统的消息推送给大家解释下MQ的意义。
三、RabbitMQ
RabbitMQ 是一个由 Erlang 语言开发的 AMQP 的开源实现。RabbitMQ是一个消息代理 - 一个消息系统的媒介。它可以为你的应用提供一个通用的消息发送和接收平台,并且保证消息在传输过程中的安全。使用消息中间件利于应用之间的解耦,生产者(客户端)无需知道消费者(服务端)的存在。而且两端可以使用不同的语言编写,大大提供了灵活性。
四、RabbitMQ的安装
【Windows安装RabbitMQ详细教程】_rabbitmq windows-CSDN博客
五、RabbitMQ的工作模型
5.1简单模式
#生产者
import pika
connection = pika.BlockingConnection(pika.ConnectionParameters('localhost'))
channel = connection.channel()
#创建队列
channel.queue_declare(queue='hello')
#往队列插入数据
channel.basic_publish(exchange='',
routing_key='hello',
body='Hello World!')
print(" [x] Sent 'Hello World!'")
# 消费者
import pika
connection = pika.BlockingConnection(pika.ConnectionParameters('localhost'))
channel = connection.channel()
channel.queue_declare(queue='hello')
#回调函数
def callback(ch, method, properties, body):
print(" [x] Received %r" % body)
#确定监听队列
channel.basic_consume(queue='hello',
auto_ack=True,#默认应答
on_message_callback=callback)
print(' [*] Waiting for messages. To exit press CTRL+C')
channel.start_consuming()
5.2RabbitMQ参数
应答参数
#生产者代码不变
#消费者
import pika
connection = pika.BlockingConnection(pika.ConnectionParameters('localhost'))
channel = connection.channel()
channel.queue_declare(queue='hello')
#回调函数
def callback(ch, method, properties, body):
print(" [x] Received %r" % body)
ch.basic_ack(delivery_tag=method.delivery_tag) #添加这一句
#确定监听队列
channel.basic_consume(queue='hello',
auto_ack=False,#改为False,手动应答
on_message_callback=callback)
print(' [*] Waiting for messages. To exit press CTRL+C')
channel.start_consuming()
默认应答的意思是当生产者向消费者发送数据的时候,假如消费者在处理数据的过程中出现了一些错误,这时候生产者队列中的数据不会进行保留,当再次运行消费者的时候,数据已经丢失。手动应答就是在处理数据过程中,如果出现了错误,生产者队列的数据不会丢失,还保留在那,直到消费者向生产者传递了信号ch.basic_ack(delivery_tag=method.delivery_tag),这时候,生产者中的数据才会删除。
分发参数
有两个消费者同时监听一个的队列。其中一个线程sleep2秒,另一个消费者线程sleep1秒,但是处理的消息是一样多。这种方式叫轮询分发不管谁忙,都不会多给消息,总是你一个我一个。想要做到公平分发(根据消费者的消费能力进行公平分发,处理快的处理的多,处理慢的处理的少;按劳分配;),必须关闭自动应答ack,改成手动应答。使用basicQos(perfetch=1)限制每次只发送不超过1条消息到同一个消费者,消费者必须手动反馈告知队列,才会发送下一个。
### 消费者
import pika
connection = pika.BlockingConnection(pika.ConnectionParameters('localhost'))
channel = connection.channel()
channel.queue_declare(queue='hello',durable=True)
#回调函数
def callback(ch, method, properties, body):
print(" [x] Received %r" % body)
#公平分发
channel.basic_qos(prefetch_count=1)
#确定监听队列
channel.basic_consume(queue='hello',
auto_ack=False,#手动应答
on_message_callback=callback)
print(' [*] Waiting for messages. To exit press CTRL+C')
channel.start_consuming()
持久化参数
持久化的意思就是把数据写入磁盘;举个例子:当生产者往队列里面插入数据的时候i,此时RabbitMQ突然崩了,而消费者一开始没有去监听这个队列,当它崩掉的时候,消费者此时去监听这个队列,这个队列是没有信息了的,因为RabbitMQ往队列插入的数据是写到内存中的,而它已经崩掉了,那就意味着数据丢失了。要解决这个问题就需要把数据写入磁盘。
#只对生产者代码进行修改,这里注意,如果之前声明过一个队列,此时就不能再次把它重新定义为持久化队列
#持久化参数解决的是RabbitMQ在运行过程中突然宕机导致数据丢失的问题
#声明queue
channel.queue_declare(queue='hello2', durable=True) # 若声hello2明过,则换一个名字
channel.basic_publish(exchange='',
routing_key='hello2',
body='Hello World!',
properties=pika.BasicProperties(
delivery_mode=2, # make message persistent
)
)
5.3交换机模式
发布订阅
发布订阅和简单的消息队列区别在于,发布订阅会将消息发送给所有的订阅者,而消息队列中的数据被消费一次便消失。所以,RabbitMQ实现发布和订阅时,会为每一个订阅者创建一个队列,而发布者发布消息时,会将消息放置在所有相关队列中。
# 生产者
import pika
connection = pika.BlockingConnection(pika.ConnectionParameters(
host='localhost'))
channel = connection.channel()
#声明交换机 名字为logs 模式为发布订阅(fanout)
channel.exchange_declare(exchange='logs',
exchange_type='fanout')
message = "info: Hello World!"
#向logs交换机插入数据
channel.basic_publish(exchange='logs',
routing_key='',
body=message)
print(" [x] Sent %r" % message)
connection.close()
# 消费者
import pika
connection = pika.BlockingConnection(pika.ConnectionParameters(host='localhost'))
channel = connection.channel()
#声明交换机 名字为logs 模式为发布订阅(fanout)
channel.exchange_declare(exchange='logs',
exchange_type='fanout')
#声明队列,名字是系统随机的,
result = channel.queue_declare("",exclusive=True)
#获取队列名字
queue_name = result.method.queue
#将指定队列绑定到logs交换机上
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)
channel.basic_consume(queue=queue_name,
auto_ack=True,
on_message_callback=callback)
channel.start_consuming()
连续启动同一份代码可能会报错,解决如下:
关键字模式
关键字模式的意思就是消费者通过绑定关键字来确定消息是否是自己需要的。之前的发布订阅模式是生产者往交换机中插入数据,而交换机则把数据插入到消费者生产的队列中,无论消息是否是消费者需要的,只要有数据都会往里面插入。而通过关键字绑定就能避开这个问题。一个消费者可以绑定多个关键字。而生产者在发送消息的时候需要指定消息类型,即routing_key。代码如下:
# 生产者
import pika
connection = pika.BlockingConnection(pika.ConnectionParameters(
host='localhost'))
channel = connection.channel()
#声明交换机 名字为logs2 模式为关键字模式(direct)
channel.exchange_declare(exchange='logs2',
exchange_type='direct')
message = "info: Hello Yuan!"
#绑定关键字routing_key='info'
channel.basic_publish(exchange='logs2',
routing_key='info',
body=message)
print(" [x] Sent %r" % message)
connection.close()
# 消费者
import pika
import sys
connection = pika.BlockingConnection(pika.ConnectionParameters(
host='localhost'))
channel = connection.channel()
channel.exchange_declare(exchange='logs2',
exchange_type='direct')
result = channel.queue_declare("",exclusive=True)
queue_name = result.method.queue
severities = sys.argv[1:]
if not severities:
sys.stderr.write("Usage: %s [info] [warning] [error]\n" % sys.argv[0])
sys.exit(1)
#绑定多个关键字
for severity in severities:
channel.queue_bind(exchange='logs2',
queue=queue_name,
routing_key=severity)
def callback(ch, method, properties, body):
print(" [x] %r" % body)
channel.basic_consume(queue=queue_name,
auto_ack=True,
on_message_callback=callback)
channel.start_consuming()
通配符
通配符交换机”与之前的路由模式相比,它将信息的传输类型的key更加细化,以“key1.key2.keyN....”的模式来指定信息传输的key的大类型和大类型下面的小类型,让消费者可以更加精细的确认自己想要获取的信息类型。而在消费者一段,不用精确的指定具体到哪一个大类型下的小类型的key,而是可以使用类似正则表达式(但与正则表达式规则完全不同)的通配符在指定一定范围或符合某一个字符串匹配规则的key,来获取想要的信息。
“通配符交换机”(Topic Exchange)将路由键和某模式进行匹配。此时队列需要绑定在一个模式上。符号“#”匹配一个或多个词,符号“*”仅匹配一个词。因此“audit.#”能够匹配到“audit.irs.corporate”,但是“audit.*”只会匹配到“audit.irs”。(这里与我们一般的正则表达式的“*”和“#”刚好相反,这里我们需要注意一下。)
下面是一个解释通配符模式交换机工作的一个样例
# 生产者
import pika
connection = pika.BlockingConnection(pika.ConnectionParameters(
host='localhost'))
channel = connection.channel()
#声明交换机 名字为logs3 模式为通配符模式(topic)
channel.exchange_declare(exchange='logs3',
exchange_type='topic')
message = "info: Hello ERU!"
channel.basic_publish(exchange='logs3',
routing_key='europe.news',
body=message)
print(" [x] Sent %r" % message)
connection.close()
# 消费者
import pika
import sys
connection = pika.BlockingConnection(pika.ConnectionParameters(
host='localhost'))
channel = connection.channel()
#声明交换机 名字为logs3 模式为通配符模式(topic)
channel.exchange_declare(exchange='logs3',
exchange_type='topic')
result = channel.queue_declare("",exclusive=True)
queue_name = result.method.queue
#通配符“#”匹配一个或多个词,符号“*”仅匹配一个词,因此“audit.#”能够匹配到“audit.irs.corporate”,但是“audit.*”只会匹配到“audit.irs”。(这里与我们一般的正则表达式的“*”和“#”刚好相反,这里我们需要注意一下。)
channel.queue_bind(exchange='logs3',
queue=queue_name,
routing_key="#.news")
print(' [*] Waiting for logs. To exit press CTRL+C')
def callback(ch, method, properties, body):
print(" [x] %r" % body)
channel.basic_consume(queue=queue_name,
auto_ack=True,
on_message_callback=callback)
channel.start_consuming()
标签:pika,exchange,python,RabbitMQ,queue,队列,connection,channel
From: https://blog.csdn.net/m0_71660867/article/details/139231927