码农知识堂 - 1000bd
  •   Python
  •   PHP
  •   JS/TS
  •   JAVA
  •   C/C++
  •   C#
  •   GO
  •   Kotlin
  •   Swift
  • RabbitMQ 入门系列:8、扩展内容:接收信息时:可否根据RoutingKey过滤监听信息,答案是不能。


    系列目录

    RabbitMQ 入门系列:1、MQ的应用场景的选择与RabbitMQ安装。

    RabbitMQ 入门系列:2、基础含义:链接、通道、队列、交换机。

    RabbitMQ 入门系列:3、基础含义:持久化、排它性、自动删除、强制性、路由键。

    RabbitMQ 入门系列:4、基础编码:官方SDK使用:链接创建、单例改造、发送消息、接收消息。

    RabbitMQ 入门系列:5、基础编码:交换机的进阶介绍及编码方式。

    RabbitMQ 入门系列:6、保障消息:不丢失:发送方、Rabbit存储端、接收方。

    RabbitMQ 入门系列:7、保障消息:不重复消费:产生消息的唯一ID。

    RabbitMQ 入门系列:8、扩展内容:接收信息时:可否根据RoutingKey过滤监听信息,答案是不能。

    RabbitMQ 入门系列:9、扩展内容:死信队列:真不适合当延时队列。

    RabbitMQ 入门系列:10、扩展内容:延时队列:延时队列插件及其有限的适用场景。

    前言:

    初看交换机的概念时,以为会有根据绑定的RoutingKey进行过滤监听的功能。

    而网上,也有一些误导人代码,连我也被误了不少时间。

    答案已经在标题上有了,下面看一些错误的演示。

    1、网上离谱的标准错误答案:DotNet版本

    1、发送方:

    复制代码
    ConnectionFactory factory = new ConnectionFactory()
    {
        HostName = "127.0.0.1",
        UserName = "guest",
        Password = "guest",
        VirtualHost = "/"
    };
    string myexchange = "myexchange";string myqueue = "myqueue";
    using (var connection = factory.CreateConnection())
    {
        var channel = connection.CreateModel();
    
        channel.ExchangeDeclare(myexchange, ExchangeType.Direct, true, false, null);
        channel.QueueDeclare(myqueue, true, false, false, null);
        channel.QueueBind(myqueue, myexchange, "log_info", null);
        channel.QueueBind(myqueue, myexchange, "log_error", null);
        for (int i = 0; i < 10; i++)
        {
            var msg = Encoding.UTF8.GetBytes($"{i},你好");
            var routeKey = i % 2 == 0 ? "log_info" : "log_error";
            channel.BasicPublish(myexchange, routingKey: routeKey, basicProperties: null, body: msg);
        }
    }
    复制代码

    向队列myqueue发送了10条数据:

     

    可以看到,队列消息里是带有Routing Key。

    然后以为可以只监听获取对应的Routing Key的数据,这种错误认知就被带出了,还演示了无效的代码:

    2、接收方:

    复制代码
    ConnectionFactory factory = new ConnectionFactory()
    {
        HostName = "127.0.0.1",
        UserName = "guest",
        Password = "guest",
        VirtualHost = "/"
    };
    string myexchange = "myexchange";
    
    string myqueue = "myqueue";
    using (var connection = factory.CreateConnection())
    {
        var channel = connection.CreateModel();
    
        channel.QueueBind(myqueue, myexchange, "log_info", null);
    
        EventingBasicConsumer consumer = new EventingBasicConsumer(channel);
    
        consumer.Received += (sender, e) =>
        {
            var msg = Encoding.UTF8.GetString(e.Body);
    
            Console.WriteLine(msg + " Routing Key :" + e.RoutingKey);
        };
    
        channel.BasicConsume(myqueue, false, consumer);
    
        Console.ReadKey();
    }
    复制代码

    天真以为绑定对应路由器就有效,运行结果仍是:

    真实的答案就只有一个:它是队列,不是数据库,无法根据条件过滤过查询。 

    2、网上容易误导的进阶式标准错误答案2:来源参考的是Java版本

     发送方保持不变,修改接收方的代码:

    复制代码
    ConnectionFactory factory = new ConnectionFactory()
    {
        HostName = "127.0.0.1",
        UserName = "guest",
        Password = "guest",
        VirtualHost = "/"
    };
    string myexchange = "myexchange";
    
    using (var connection = factory.CreateConnection())
    {
        var channel = connection.CreateModel();
    
        string name = channel.QueueDeclare().QueueName;
        channel.QueueBind(name, myexchange, "log_info", null);
    
        EventingBasicConsumer consumer = new EventingBasicConsumer(channel);
    
        consumer.Received += (sender, e) =>
        {
            var msg = Encoding.UTF8.GetString(e.Body);
    
            Console.WriteLine(msg + " Routing Key :" + e.RoutingKey);
        };
    
        channel.BasicConsume(name, false, consumer);
    
        Console.ReadKey();
    }
    复制代码

    重新定义临时队列,然后绑定了临时队列的名称,开启监听,然后让发送方再次发送数据:

    右边发送了10条,左边根据过滤条件,收到了5条,看起来好像达到效果了。

    表现上很正确,这就很TMD的误导人了。

    下面看看这种是什么情况:

     

    实际的情况是:

    1、接收方:创建了临时队列,同时绑定log_info,监听的是临时队列。
    
    2、发送方:即发送到myqueue,对log_info的数据,再copy一份发送到临时队列。

    这就很离谱了,会造成以下几个问题:

    1、只能收到监听开启之后发送的数据,对myqueue之前的数据是获取不到的。
    
    也即是说,仅有双方同时在线,发送时的数据才能收到,其余条件的数据都会丢失。
    
    2、myqueue还是存有一份信息,不消费,攒着留着过年吗?

    第2个条件还能通过设置过期解决,第1个条件,都不知道什么应用场景才适合,但肯定不属于MQ的应用场景。

    总结:

    还是一句话:它是队列,不是数据库,无法根据条件过滤过查询。 

  • 相关阅读:
    基于Django+Vue开发的社区疫情管理系统(附源码)
    Spring(五)Spring 管理第三方Bean和核心容器
    LeetCode 460. LFU 缓存
    AI视频智能分析引领智慧园区升级:EasyCVR智慧园区视频管理方案
    线程与进程
    记一次 HTTPS 抓包分析和 SNI 的思考
    如果使用pprof来进行性能的观测和优化
    【CVPR2020】DEF:Seeing Through Fog Without Seeing Fog论文阅读分析与总结
    技术与产品:工业革命与时代进步的缩影
    【Redis】简单实现分布式锁
  • 原文地址:https://www.cnblogs.com/cyq1162/p/16607863.html
  • 最新文章
  • 攻防演习之三天拿下官网站群
    数据安全治理学习——前期安全规划和安全管理体系建设
    企业安全 | 企业内一次钓鱼演练准备过程
    内网渗透测试 | Kerberos协议及其部分攻击手法
    0day的产生 | 不懂代码的"代码审计"
    安装scrcpy-client模块av模块异常,环境问题解决方案
    leetcode hot100【LeetCode 279. 完全平方数】java实现
    OpenWrt下安装Mosquitto
    AnatoMask论文汇总
    【AI日记】24.11.01 LangChain、openai api和github copilot
  • 热门文章
  • 十款代码表白小特效 一个比一个浪漫 赶紧收藏起来吧!!!
    奉劝各位学弟学妹们,该打造你的技术影响力了!
    五年了,我在 CSDN 的两个一百万。
    Java俄罗斯方块,老程序员花了一个周末,连接中学年代!
    面试官都震惊,你这网络基础可以啊!
    你真的会用百度吗?我不信 — 那些不为人知的搜索引擎语法
    心情不好的时候,用 Python 画棵樱花树送给自己吧
    通宵一晚做出来的一款类似CS的第一人称射击游戏Demo!原来做游戏也不是很难,连憨憨学妹都学会了!
    13 万字 C 语言从入门到精通保姆级教程2021 年版
    10行代码集2000张美女图,Python爬虫120例,再上征途
Copyright © 2022 侵权请联系2656653265@qq.com    京ICP备2022015340号-1
正则表达式工具 cron表达式工具 密码生成工具

京公网安备 11010502049817号