rabbitmq安装和thinkphp6简单操作思路

news2025/1/22 9:23:15

1、安装rabbitmq

2、crontab定时检测rabbtimq状态

2、使用thinphp6.0框架rabbitmq示例,supervisor守护消费者

3、RabbitMQ有四种交换机类型

rabbitmq组成部分如下:

Broker:消息队列服务进程,此进程包括两个部分:Exchange和Queue。
Exchange:消息队列交换机,按一定的规则将消息路由转发到某个队列对消息进行过虑。
Queue:消息队列,存储消息的队列,消息到达队列并转发给指定的消费方。
Producer:消息生产者,即生产方客户端,生产方客户端将消息发送到MQ。
Consumer:消息消费者,即消费方客户端,接收MQ转发的消息。

消息发布接收流程:

—–发送消息—–
1、生产者和Broker建立TCP连接。
2、生产者和Broker建立通道。
3、生产者通过通道消息发送给Broker,由Exchange将消息进行转发。
4、Exchange将消息转发到指定的Queue(队列)

—-接收消息—–

1、消费者和Broker建立TCP连接
2、消费者和Broker建立通道
3、消费者监听指定的Queue(队列)
4、当有消息到达Queue时Broker默认将消息推送给消费者。
5、消费者接收到消息。
6、消费者应使用supervisor进行监控保持在线

1、方式一:docker安装RabbitMQ

查看仓库里的RabbitMQ

 [root@localhost ~]# docker search rabbitmq

安装RabbitMQ

[root@localhost ~]#  docker pull rabbitmq

这里是直接安装最新的,如果需要安装其他版本在rabbitmq后面跟上版本号即可

启动RabbitMQ

 [root@localhost ~]# docker run -d --hostname my-rabbit --name rabbit -p 15672:15672 -p 5672:5672 rabbitmq

安装插件

1. 先执行docker ps 拿到当前的镜像ID

[root@localhost ~]#  docker ps

2. 进入容器

[root@localhost ~]# docker exec -it 镜像ID  /bin/bash

3. 安装web页面插件

 [root@localhost ~]# rabbitmq-plugins enable rabbitmq_management

2、方式二:CentOS6.9下安装rabbitmq消息队列

安装如下步骤:

首先安装erlang

[root@localhost ~]# yum install erlang

安装rabbitmq rpm包

[root@localhost ~]# wget http://www.rabbitmq.com/releases/rabbitmq-server/v3.5.0/rabbitmq-server-3.5.0-1.noarch.rpm
[root@localhost ~]# rpm -ivh rabbitmq-server-3.5.0-1.noarch.rpm

启动rabbitmq

[root@localhost ~]# service start rabbitmq-server

注:如果启动失败,报错如下:

则修改hosts,添加hostname:

[root@localhost ~]# vi /etc/hosts

添加hostname:

再次重启即可:

打开5672端口(注意 15672,5672端口防火墙之类的问题需要开启一下

[root@localhost ~]#  /sbin/iptables -I INPUT -p tcp --dport 5672 -j ACCEPT   
[root@localhost ~]#  /etc/rc.d/init.d/iptables save 
[root@localhost ~]#  /etc/init.d/iptables restart

启用web页面插件

[root@localhost ~]#  rabbitmq-plugins enable rabbitmq_management

重启rabbitmq

[root@localhost ~]# systemctl start rabbitmq-server

web访问: ip地址可以通过ip add 或者 ifconfig查看

用户名和密码:guest guest

无法登录请使用如下:

[root@localhost ~]# vim /etc/rabbitmq/rabbitmq.config

写入如下信息并保存:

[{rabbit, [{loopback_users, []}]}].

再重启:

再次访问即可:

服务启动
[root@localhost ~]# systemctl start rabbitmq-server.service
看看是否启动成功
[root@localhost ~]# rabbitmqctl status
设置开机自启
[root@localhost ~]# [root@localhost ~]# chkconfig rabbitmq-server on
添加到启动项并设置开机自启
[root@localhost ~]# systemctl enable rabbitmq-server.service
开启管理界面
[root@localhost ~]# rabbitmq-plugins enable rabbitmq_management
添加账号
[root@localhost ~]# rabbitmqctl add_user abc 123456
设置用户角色
[root@localhost ~]# rabbitmqctl set_user_tags abc administrator
设置用户权限
[root@localhost ~]# rabbitmqctl set_permissions -p "/" abc ".*" ".*" ".*"
查看用户和角色 需要启动服务
[root@localhost ~]# rabbitmqctl list_users
删除角色
[root@localhost ~]# rabbitmqctl delete_user Username

2、crontab定时检测rabbtimq状态(应使用supervisor实时监听)

1)crontab定时任务(每分钟检查运行脚本【分,时,日,月,周】)

[root@localhost ~]# crontab -e

...
*/1 * * * * /bin/bash /usr/local/sbin/rabbitmq.sh

2)rabbitmq启动脚本(通过获取pgrep获取rabbitmq-server进程号再进行判断)

[root@localhost ~]# cd /usr/local/sbin
[root@localhost ~]# vim rabbitmq.sh
#!/bin/bash  
pgrep -x rabbitmq-server &> /dev/null

if [ $? -ne 0 ]

then

echo "At time: `date` :rabbitmq error stop .">> /var/log/rabbitmqCheck.log

/etc/init.d/rabbitmq-server start
#echo "At time: `date` :rabbitmq server is stop."  

else
echo "rabbitmq server is running ." >> /var/log/rabbitmqCheck.log  
fi

测试:

1)停止rabbitmq【Active:inactive(dead)】

[root@localhost sbin]# /etc/init.d/rabbitmq-server stop
Stopping rabbitmq-server (via systemctl):                  [  OK  ]
[root@localhost sbin]# 
[root@localhost sbin]# systemctl status rabbitmq-server
● rabbitmq-server.service - LSB: Enable AMQP service provided by RabbitMQ broker
   Loaded: loaded (/etc/rc.d/init.d/rabbitmq-server; bad; vendor preset: disabled)
   Active: inactive (dead) since Wed 2021-01-13 18:01:33 CST; 46s ago
     Docs: man:systemd-sysv-generator(8)
  Process: 14618 ExecStop=/etc/rc.d/init.d/rabbitmq-server stop (code=exited, status=0/SUCCESS)
  Process: 14346 ExecStart=/etc/rc.d/init.d/rabbitmq-server start (code=exited, status=0/SUCCESS)

Jan 13 18:01:02 localhost.localdomain su[14426]: (to rabbitmq) root on none
Jan 13 18:01:02 localhost.localdomain su[14421]: (to rabbitmq) root on none
Jan 13 18:01:05 localhost.localdomain rabbitmq-server[14346]: Starting rabbitmq-server: SUCCESS
Jan 13 18:01:05 localhost.localdomain rabbitmq-server[14346]: rabbitmq-server.
Jan 13 18:01:05 localhost.localdomain systemd[1]: Started LSB: Enable AMQP service provided by RabbitMQ broker.
Jan 13 18:01:29 localhost.localdomain systemd[1]: Stopping LSB: Enable AMQP service provided by RabbitMQ broker...
Jan 13 18:01:29 localhost.localdomain su[14623]: (to rabbitmq) root on none
Jan 13 18:01:30 localhost.localdomain su[14684]: (to rabbitmq) root on none
Jan 13 18:01:33 localhost.localdomain rabbitmq-server[14618]: Stopping rabbitmq-server: rabbitmq-server.
Jan 13 18:01:33 localhost.localdomain systemd[1]: Stopped LSB: Enable AMQP service provided by RabbitMQ broker.
[root@localhost sbin]# 

2)隔一分钟再次查看rabbitmq状态【Active:active running】

[root@localhost sbin]# systemctl status rabbitmq-server

● rabbitmq-server.service - LSB: Enable AMQP service provided by RabbitMQ broker
   Loaded: loaded (/etc/rc.d/init.d/rabbitmq-server; bad; vendor preset: disabled)
   Active: active (running) since Wed 2021-01-13 18:03:04 CST; 35s ago
     Docs: man:systemd-sysv-generator(8)
  Process: 14618 ExecStop=/etc/rc.d/init.d/rabbitmq-server stop (code=exited, status=0/SUCCESS)
  Process: 14917 ExecStart=/etc/rc.d/init.d/rabbitmq-server start (code=exited, status=0/SUCCESS)
    Tasks: 2
   Memory: 1.4M
   CGroup: /system.slice/rabbitmq-server.service
           ├─14990 /bin/sh /etc/rc.d/init.d/rabbitmq-server start
           └─14995 /bin/bash -c ulimit -S -c 0 >/dev/null 2>&1 ; /usr/sbin/rabbitmq-server

Jan 13 18:03:01 localhost.localdomain systemd[1]: Starting LSB: Enable AMQP service provided by RabbitMQ broker...
Jan 13 18:03:01 localhost.localdomain su[14921]: (to rabbitmq) root on none
Jan 13 18:03:02 localhost.localdomain su[14998]: (to rabbitmq) root on none
Jan 13 18:03:02 localhost.localdomain su[14991]: (to rabbitmq) root on none
Jan 13 18:03:04 localhost.localdomain rabbitmq-server[14917]: Starting rabbitmq-server: SUCCESS
Jan 13 18:03:04 localhost.localdomain systemd[1]: Started LSB: Enable AMQP service provided by RabbitMQ broker.
Jan 13 18:03:04 localhost.localdomain rabbitmq-server[14917]: rabbitmq-server.
You have new mail in /var/spool/mail/root
[root@localhost sbin]# 

方式三:docker-compose.yml安装

 version: '3.1'
services:
 rabbitmq:
  hostname: rabbitmq
  image: rabbitmq:3.9.12-management
  container_name: rabbitmq
  privileged: true
  ports:
   - 15672:15672
   - 5672:5672
  volumes:
   - /etc/localtime:/etc/localtime
   - ./data:/var/lib/rabbitmq
  environment:
   TZ: Asia/Shanghai
   RABBITMQ_DEFAULT_USER: admin
   RABBITMQ_DEFAULT_PASS: 123456

3、thinkphp6.0使用composer安装rabbitmq安装包

本次需要修改的文件有

Comer.php 使用php think生产的命令行
Index.php 测试类
MqConsumer.php 消费者
MqProducer.php 生产者
Console.php 命令行配置项
rabbitmq.php 消息队列配置项

1、安装amqplib

进入到tp6项目根目录安装rabbitmq包,如需忽略版本安装 --ignore-platform-reqs

$composer require --ignore-platform-reqs php-amqplib/php-amqplib

2、在config文件夹下添加rabbitmq.php配置文件

<?php
// 示例配置文件
return [
 
 # 连接信息
 'AMQP' => [
 'host' => '192.168.1.130',   //连接rabbitmq,此为安装rabbitmq服务器
 'port'=>'5672',
 'login'=>'guest',
 'password'=>'guest',
 'vhost'=>'/'
 ],
 # 队列
 'direct_queue' => [
 'exchange_name' => 'direct_exchange',
 'exchange_type'=>'direct',#直连模式
 'queue_name' => 'direct_queue',
 'route_key' => 'direct_roteking',
 'consumer_tag' => 'direct'
 ]
];

3、编写生产者代码

<?php
namespace app\controller;

use app\BaseController;
use PhpAmqpLib\Connection\AMQPStreamConnection;
use PhpAmqpLib\Message\AMQPMessage;
use think\facade\Log;

class MqProducer
{
    public static function pushMessage($data)
    {
        $param = config('rabbitmq.AMQP');
        $amqpDetail = config('rabbitmq.direct_queue');
        $connection = new AMQPStreamConnection(
            $param['host'],
            $param['port'],
            $param['login'],
            $param['password'],
            $param['vhost']
        );
        $channel = $connection->channel();
        /*
         * 创建队列(Queue)
         * name: hello         // 队列名称
         * passive: false      // 如果设置true存在则返回OK,否则就报错。设置false存在返回OK,不存在则自动创建
         * durable: true       // 是否持久化,设置false是存放到内存中RabbitMQ重启后会丢失,
         *                        设置true则代表是一个持久的队列,服务重启之后也会存在,因为服务会把持久化的Queue存放在硬盘上,当服务重启的时候,会重新加载之前被持久化的Queue
         * exclusive: false    // 是否排他,指定该选项为true则队列只对当前连接有效,连接断开后自动删除
         *  auto_delete: false // 是否自动删除,当最后一个消费者断开连接之后队列是否自动被删除
         */
        $channel->queue_declare($amqpDetail['queue_name'], false, true, false, false);

        /*
         * 创建交换机(Exchange)
         * name: vckai_exchange// 交换机名称
         * type: direct        // 交换机类型,分别为direct/fanout/topic,参考另外文章的Exchange Type说明。
         * passive: false      // 如果设置true存在则返回OK,否则就报错。设置false存在返回OK,不存在则自动创建
         * durable: false      // 是否持久化,设置false是存放到内存中的,RabbitMQ重启后会丢失
         * auto_delete: false  // 是否自动删除,当最后一个消费者断开连接之后队列是否自动被删除
         */
        $channel->exchange_declare($amqpDetail['exchange_name'], $amqpDetail['exchange_type'], false, true, false);

        /*
         * 绑定队列和交换机
         * @param string $queue 队列名称
         * @param string $exchange  交换器名称
         * @param string $routing_key   路由key
         * @param bool $nowait
         * @param array $arguments
         * @param int|null $ticket
         * @throws \PhpAmqpLib\Exception\AMQPTimeoutException if the specified operation timeout was exceeded
         * @return mixed|null
         */
        $channel->queue_bind($amqpDetail['queue_name'], $amqpDetail['exchange_name'], $amqpDetail['route_key']);

        /*
             $messageBody:消息体
             content_type:消息的类型 可以不指定
             delivery_mode:消息持久化最关键的参数
             AMQPMessage::DELIVERY_MODE_NON_PERSISTENT = 1; 不持久化
             AMQPMessage::DELIVERY_MODE_PERSISTENT = 2; 持久化
         */

        //将要发送数据变为json字符串
        $messageBody = json_encode($data);
        /*
         * 创建AMQP消息类型
         * $messageBody:消息体
         * delivery_mode 消息是否持久化
         *      AMQPMessage::DELIVERY_MODE_NON_PERSISTENT = 1; 不持久化
         *      AMQPMessage::DELIVERY_MODE_PERSISTENT = 2; 持久化
         */
        $message = new AMQPMessage($messageBody, array('content_type' => 'text/plain', 'delivery_mode' => AMQPMessage::DELIVERY_MODE_PERSISTENT));

        /*
         * 发送消息
         * msg       // AMQP消息内容
         * exchange  // 交换机名称
         * routing key     // 路由键名称
         */
        $channel->basic_publish($message, $amqpDetail['exchange_name'],$amqpDetail['route_key']);
        $channel->close();
        $connection->close();
        echo  "ok";
    }
}

4、消费者代码

<?php
namespace app\controller;

use app\BaseController;
use PhpAmqpLib\Connection\AMQPStreamConnection;
use think\Controller;
use think\facade\Log;

class MqConsumer
{
    /**
     * 消费端 消费端需要保持运行状态实现方式
     * 1 linux上写定时任务每隔5分钟运行下该脚本,保证访问服务器的ip比较平缓,不至于崩溃
     * 2 nohup php index.php index/Message_Consume/start &  用nohup命令后台运行该脚本
     * 3
     **/
    function shutdown($channel, $connection)
    {
        $channel->close();
        $connection->close();
        Log::write("closed",3);
    }

    //消息处理
    function process_message($message)
    {

        //休眠两秒
        //sleep(2);
        echo  $message->body."\n";
        //自定义日志为rabbitmq-consumer
        Log::write($message->body,'rabbitmq-consumer');
        //[2021-01-14T16:14:17+08:00][rabbitmq-consumer] {"time":1610612057,"order":85}

        //手动发送ack
        $message->delivery_info['channel']->basic_ack($message->delivery_info['delivery_tag']);
        // Send a message with the string "quit" to cancel the consumer.
        if ($message->body === 'quit') {
            $message->delivery_info['channel']->basic_cancel($message->delivery_info['consumer_tag']);
        }
    }

    /**
     * 启动
     * @return \think\Response
     */
    public function start()
    {
        $param = config('rabbitmq.AMQP');
        $amqpDetail = config('rabbitmq.direct_queue');
        $connection = new AMQPStreamConnection(
            $param['host'],
            $param['port'],
            $param['login'],
            $param['password'],
            $param['vhost']
        );

        /*
         * 创建通道
         */
        $channel = $connection->channel();
        /*
         * 设置消费者(Consumer)客户端同时只处理一条队列
         * 这样是告诉RabbitMQ,再同一时刻,不要发送超过1条消息给一个消费者(Consumer),
         * 直到它已经处理了上一条消息并且作出了响应。这样,RabbitMQ就会把消息分发给下一个空闲的消费者(Consumer)。
         */
        $channel->basic_qos(0, 1, false);
        /*
         * 同样是创建路由和队列,以及绑定路由队列,注意要跟publisher的一致
         * 这里其实可以不用,但是为了防止队列没有被创建所以做的容错处理
         */
        $channel->queue_declare($amqpDetail['queue_name'], false, true, false, false);

        $channel->exchange_declare($amqpDetail['exchange_name'], $amqpDetail['exchange_type'], false, true, false);

        $channel->queue_bind($amqpDetail['queue_name'], $amqpDetail['exchange_name'],$amqpDetail['route_key']);

        /*
            queue: 从哪里获取消息的队列
            consumer_tag: 消费者标识符,用于区分多个客户端
            no_local: 不接收此使用者发布的消息
            no_ack: 设置为true,则使用者将使用自动确认模式。详情请参见.
                        自动ACK:消息一旦被接收,消费者自动发送ACK
                        手动ACK:消息接收后,不会发送ACK,需要手动调用
            exclusive:是否排他,即这个队列只能由一个消费者消费。适用于任务不允许进行并发处理的情况下
            nowait: 不返回执行结果,但是如果排他开启的话,则必须需要等待结果的,如果两个一起开就会报错
            callback: :回调逻辑处理函数,PHP回调 array($this, 'process_message') 调用本对象的process_message方法
        */
        $channel->basic_consume($amqpDetail['queue_name'], $amqpDetail['consumer_tag'], false, false, false, false, array($this, 'process_message'));

        register_shutdown_function(array($this, 'shutdown'), $channel, $connection);
        // 阻塞队列监听事件
        while (count($channel->callbacks)) {
            $channel->wait();
        }
    }
}

5、编写测试代码Index.php

<?php
namespace app\controller;

use app\BaseController;
use app\controller\MqProducer;

class Index
{
    public function index()
    {
        echo "tp6";die;
    }

    public function hello($name = 'ThinkPHP6')
    {
        return 'hello,' . $name;
    }

    public function send()
    {

//        for($i=0; $i<5; $i++){
            $consumer = new MqProducer();//生产者
            $data = [
              'time'=>time(),
              'order'=> rand(1, 100),
            ];
            $consumer->pushMessage($data);

//            sleep(1);
//        }
    }
}

 

6、项目更目录使用php think生成指令

1)进入到项目目录:php -v 查看是否配置环境,否则配置php系统环境

Windows7下的php环境配置教程_php技巧_脚本之家​www.jb51.net/article/61507.htm正在上传…重新上传取消

2)执行:php think make:command Comer Comer 会在command目录下生成Comer .php文件

3)在config/console.php添加指令

<?php
// +----------------------------------------------------------------------
// | 控制台配置
// +----------------------------------------------------------------------
return [
    // 指令定义
    'commands' => [
        // consumer是app\Command\Consumer文件中自定义命令行的名字不能使用
        'comer' => 'app\command\Comer',
    ],
];

4.修改指令文件

<?php
declare (strict_types = 1);

namespace app\command;

use app\controller\MqConsumer;
use think\console\Command;
use think\console\Input;
use think\console\input\Argument;
use think\console\input\Option;
use think\console\Output;

class Comer extends Command
{
    protected function configure()
    {
        // 指令配置
        $this->setName('Comer')
            ->setDescription('the Comer command');
    }

    protected function execute(Input $input, Output $output)
    {
        // 指令输出
//        $output->writeln('Comer');
        $consumer = new MqConsumer(); //消费者
        $consumer->start();  //启动
    }
}

7、测试

注释:我的域名为:www.tp6.net

步骤:

(1)启动监听指令(消费者消费)
(2)生产者发送消息
(3)查看rabbitmq管理界面是否有消息
(4)查看消费者已消费消息

1)在项目根目录指令监听 php think comer

2)发送消息

http://www.tp6.net/index.php/index/send

3)rabbitmq管理界面

4)查看消费者已消费消息

8、使用supervisor监控消费者实时在线

安装supervisor:

监听守护【rabbitmq消费者】配置文件如下,然后保存退出

[user@localhost  tp6.com]$ cd /etc/supervisord.d/
[user@localhost supervisord.d]$ sudo vim tp_amqp.ini
[program:tp_amqp]
directory = /www/wwwroot/www.tp6.com                           ;启动目录
command = /www/server/php/72/bin/php think comer       ;启动命令
autostart = true                ;在supervisord启动的时候也启动
startsecs = 5                   ;启动5秒后没有异常退出,就当作已经正常启动了
autorestart = true              ;程序异常退出后自动重启
startretries = 3                ;启动失败自动重试次数,默认是3
user = root                     ;哪个用户启动
redirect_stderr = true          ;把stderr重定向到stdout,默认false
stdout_logfile_maxbytes = 20MB  ;stdout日志文件大小,默认50MB
stdout_logfile_backups = 20     ;stdout日志文件备份数
stdout_logfile = /usr/log/tp_amqp.log
;stdout日志文件,需要手动创建/usr/log/tp_amqp.log

4、启动守护进程

更新配置文件:supervisorctl update
启动进程:sudo supervisorctl start tp_amqp

通过ps查看守护的命令:

[user@localhost www.tp6.com]$ sudo supervisorctl status tp_amqp
tp_amqp                          RUNNING   pid 16595, uptime 0:01:27
[user@localhost www.tp6.com]$ ps afx | grep php
 2364 ?        Ss     0:00 php-fpm: master process (/www/server/php/72/etc/php-fpm.conf)
 2365 ?        S      0:00  \_ php-fpm: pool www
 2366 ?        S      0:00  \_ php-fpm: pool www
 2367 ?        S      0:00  \_ php-fpm: pool www
 2368 ?        S      0:00  \_ php-fpm: pool www
 2369 ?        S      0:00  \_ php-fpm: pool www
14478 ?        S      0:00  \_ php-fpm: pool www
16746 pts/1    S+     0:00      \_ grep --color=auto php
16595 ?        S      0:00  \_ /www/server/php/72/bin/php think comer
[user@localhost www.tp6.com]$ sudo kill 16595
[user@localhost www.tp6.com]$ ps afx | grep php
 2364 ?        Ss     0:00 php-fpm: master process (/www/server/php/72/etc/php-fpm.conf)
 2365 ?        S      0:00  \_ php-fpm: pool www
 2366 ?        S      0:00  \_ php-fpm: pool www
 2367 ?        S      0:00  \_ php-fpm: pool www
 2368 ?        S      0:00  \_ php-fpm: pool www
 2369 ?        S      0:00  \_ php-fpm: pool www
14478 ?        S      0:00  \_ php-fpm: pool www
16823 pts/1    S+     0:00      \_ grep --color=auto php
16821 ?        S      0:00  \_ /www/server/php/72/bin/php think comer
[user@localhost www.tp6.com]$ 

简单描述一下RabbitMQ中的几个关键的概念:

Broker:可以简单的理解为安装了RabbitMQ服务的这台机器就可以称为中间人

Exchange:交换机,消息经由它,通过路由键来判断并决定把消息投递给哪个队列,它类似于一个路由器的角色

Queue:队列,最终将消息投递到队列中,由消费端监听队列进行消费

Binding:绑定关系,需要给交换机绑定队列,绑定时需要给一个路由键

Routingkey:路由键,交换机和队列进行绑定时,需要指定路由键或通配符路由键。

交换机根据路由键来决定消息投递到哪个或哪些队列

大致流程:使用RabbitMQ前,首先需要根据业务来创建交换机和队列,创建完成后需要给交换机绑定队列(交换机和队列可以是多对多的关系),绑定队列时要指定具体的路由键或者通配符路由键当生产者发送一条消息的时候,需要指定交换机和路由键,消息到达Broker后先转给刚才指定的交换机,交换机再根据路由键来决定把消息投递给与自己绑定的哪一个或哪一些队列,最后再由消费端来监听这些队列,消费处理对应的消息

最新版本的RabbitMQ有四种交换机类型,分别是:Direct exchange、Fanout exchange、Topic exchange、Headers exchange

1、Direct exchange---直接类型交换机

要求消息带的路由键和绑定的路由键完全匹配,这是一个完整的匹配。

2、Fanout Exchange---扇出类型交换机

只需要简单的将队列绑定到该类型交换机上,该类型的交换机绑定队列时可以不指定路由键(Routingkey)

当消息发送给该交换机后,它会将消息投递给与该交换机绑定的所有队列

很像广播,每台子网内的机器都会获得一份消息,Fanout交换机转发消息是最快的

3、Topic Exchange---主题类型交换机

将路由键和某模式进行匹配。此时队列需要绑定某一个模式上。符号#匹配0个或多个单词,符号 *匹配一个单词。

4、Headers Exchanges

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/371278.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

代码随想录算法训练营day42 |动态规划之背包问题 11.分割等和子集 1049. 最后一块石头的重量 II 494. 目标和

day4211.分割等和子集1.dp数组的含义2.确定递推公式3.dp数组初始化4.确定遍历顺序5.打印dp数组1049. 最后一块石头的重量 II1.确定dp数组以及下标的含义2.确定递推公式3.dp数组如何初始化4.确定遍历顺序5.举例推导dp数组494. 目标和1.确定dp数组以及下标的含义2.确定递推公式3.…

Bootstrap系列之Flex布局

文章目录Bootstrap中的Flexd-flex与d-inline-flex也存在响应式变化flex水平布局flex垂直布局flex水平与垂直也存在响应式变化内容排列&#xff08;justify-content响应式变化也存在于这里sm&#xff0c;md&#xff0c;lg&#xff0c;xl&#xff09;子元素对齐方式Align items&a…

Oracle-00-卸载篇

这里给出企业级的Oracle 10g的卸教程,新安装的19c并没有正经去做卸载的操作,为了后面教程的进度,这里就先借用下10g,如果有需要会重新更新19c的卸载教程 windows服务中将Oracle所有服务全部停掉 选中Oracle - OraDb10g_home2->Oracle Installation Products->Univers…

文化:你所在的团队,有多少人敢讲真话?

你好&#xff0c;我是叶芊。 今天我们要讨论的话题是文化&#xff0c;说“文化”这个词你可能会觉得很虚&#xff0c;那我们换个词——“做事风格”&#xff0c;这就和你们团队平时的协作习惯密切相关了。 做事风格&#xff0c;往小了讲&#xff0c;会影响团队成员对开会的认知…

CV——day78 读论文:通过静态背景构建扩展低通道路边雷达的探测距离(目标是规避风险)

Extending the Detection Range for Low-Channel Roadside LiDAR by Static Background Construction 通过静态背景构建扩展低通道路边雷达的探测距离I. INTRODUCTIONII. RELATED WORKA. LiDAR-Based 3-D Vehicle and Road User DetectionB. LiDAR Data Background FilteringC.…

数据结构与算法之打家劫舍(一)动态规划思想

动态规划里面一部题目打家劫舍是一类经典的算法题目之一&#xff0c;他有各种各样的变式&#xff0c;这一篇文章和大家分享一下打家劫舍最基础的一道题目&#xff0c;掌握这一道题目&#xff0c;为下一道题目打下基础。我们直接进入正题。一.题目大家如果刚接触这样的题目&…

Linux小黑板(9):共享内存

"My poor lost soul"上章花了不少的篇幅讲了讲基于管道((匿名、命名))技术实现的进程间通信。进程为什么需要通信&#xff1f;目的是为了完成进程间的"协同",提高处理数据的能力、优化业务逻辑的实现等等&#xff0c;在linux中我们已经谈过了一个通信的大类…

数据库的基本查询

注意&#xff1a;LIMIT的两个参数&#xff0c;第一个是起始位置&#xff0c;第二个是一次查询到多少页。注意&#xff1a;什么类型的数字都是可以排序的。日期的降序是从现在到以前&#xff0c;MySQL ENUM值如何排序&#xff1f;在MYSQL中&#xff0c;我们知道每个ENUM值都与一…

安装MySQL数据库

安装MySQL数据库 获取软件&#xff1a;https://dev.mysql.com/downloads/mysql/ 下载完成后进行解压操作 若安装目录里没有my.ini配置文件&#xff0c;则需要新建一个my.ini的配置文件。 编辑my.ini配置文件&#xff0c;将配置文件中的内容修改成下面内容 [client] # 设置…

基于企业微信应用消息的每日早安推送

基于企业微信应用消息的每日早安推送 第一步&#xff1a;注册企业微信 企业微信注册地址&#xff1a;https://work.weixin.qq.com/wework_admin/register_wx 按照正常流程填写信息即可&#xff0c;个人也可以注册企业微信&#xff0c;不需要公司 注册完成后&#xff0c;登录…

户籍管理系统测试用例

目录 一、根据页面的不同分别设计测试用例 登录页面 用户信息列表 用户编辑页面 用户更新页面 二、根据目的不同分别设计测试用例 一、根据页面的不同分别设计测试用例 上图是针对一个网站的测试&#xff0c;按照页面的不同分别来设计对应的测试用例。 登录页面 用户信息列…

[MySQL]MySQL数据类型

文章目录数据类型分类数值类型tinyint类型bit类型float类型decimal类型字符串类型char类型varchar类型char和varchar对比日期和时间类型enum和set类型数据类型分类 MySQL中&#xff0c;支持各种各样的类型&#xff0c;比如表示数值的整型浮点型&#xff0c;文本、二进制类型、…

【密码学】 一篇文章讲透数字签名

【密码学】 一篇文章讲透数字签名 数字签名介绍 数字签名&#xff08;又称公钥数字签名&#xff09;是只有信息的发送者才能产生的别人无法伪造的一段数字串&#xff0c;这段数字串同时也是对信息的发送者发送信息真实性的一个有效证明。它是一种类似写在纸上的普通的物理签名…

问题解决:Excel中依据某一列数据进行匹配

问题描述Excel处理时&#xff0c;常常需要从一个大表里&#xff0c;按照条件提取子集。需要我们按照某一序列为标准&#xff0c;匹配筛选出有效信息&#xff0c;案例如下&#xff1a;依据名称匹配销售额。解决方法使用函数&#xff1a;VLOOKUP(lookup_value,table_array,col_in…

Elasticsearch7.8.0版本进阶——自定义分析器

目录一、自定义分析器的概述二、自定义的分析器的测试示例一、自定义分析器的概述 Elasticsearch 带有一些现成的分析器&#xff0c;然而在分析器上 Elasticsearch 真正的强大之 处在于&#xff0c;你可以通过在一个适合你的特定数据的设置之中组合字符过滤器、分词器、词汇单 …

刚来的薪资20k,是我的2倍,我是真的卷不过,真的太变态了

在这个行业爬摸滚打5年了&#xff0c;从最开始点点点的功能测试到现在到现在成为高级测试&#xff0c;工资也翻了几倍&#xff0c;简单的说几句吧 改变的开始 之所以改变的原因很简单&#xff0c;我快被新来的卷死了&#xff0c;新来的本科是某211的&#xff0c;干劲十足&…

Python为CANoe工程添加/删除DBC文件

前面文章我们对于通过COM来实现打开CANoe、导入CANoe配置工程、导入执行文件、启动CANoe软件和执行脚本;但是这只能完成最基本的功能调用,在实际得到使用过程中,特别是各家在推的CI/CD以及平台化,仅仅是实现这些功能是完全不够用的;比如dbc的添加和删除,这是我们非常必要…

【洛谷 P1563】[NOIP2016 提高组] 玩具谜题(模拟+结构体数组+指针)

[NOIP2016 提高组] 玩具谜题 题目背景 NOIP2016 提高组 D1T1 题目描述 小南有一套可爱的玩具小人, 它们各有不同的职业。 有一天, 这些玩具小人把小南的眼镜藏了起来。 小南发现玩具小人们围成了一个圈,它们有的面朝圈内,有的面朝圈外。如下图: 这时 singer 告诉小南一个谜…

Python变量的定义和使用

定义&#xff1a;变量就是计算机内存中存储某些数据的位置的名称 形象理解变量就是一个存放东西的容器&#xff0c;该容器的名字就叫做变量&#xff0c;容器存放的东西就是变量的值 变量的组成&#xff1a; 标识&#xff1a;标识对象所储存的内存地址&#xff0c;使用内置函数i…

Redis的持久化方式

Redis支持两种方式的持久化&#xff0c;一种是RDB方式、另一种是AOF&#xff08;append-only-file&#xff09;方式&#xff0c;两种持久化方式可以单独使用其中一种&#xff0c;也可以将这两种方式结合使用。 •RDB&#xff1a;根据指定的规则“定时”将内存中的数据存储在硬…