繁体   English   中英

如何在Spring AMQP中以编程方式将异步使用方订阅到队列?

[英]How to programmatically subscribe an async consumer to a queue in Spring AMQP?

Spring AMQP文档仅显示如何使用~MessageListenerContainer在应用程序初始化时以编程方式订阅队列。

现在,代码如下所示:

public void subscribe(EventType eventType, Object consumer) {
        Assert.notNull(eventType);
        Assert.notNull(eventType.toString());
        Assert.isTrue(!eventType.toString().isEmpty());

        Queue queue=new Queue("", false, true, true);
        Map<String, Exchange> beanMap=context.getBeansOfType(Exchange.class);
        if(beanMap!=null&&!beanMap.isEmpty()){
            Exchange exchange=null;
            boolean found=false;
            for(String key : beanMap.keySet()) {
                exchange=beanMap.get(key);
                if(getExchangeName(eventType.toString()).equals(exchange.getName())){
                    found=true;
                    break;
                }
            }
            if(found){
                amqpAdmin.declareQueue(queue);
                amqpAdmin.declareBinding(new Binding(queue.getName(), Binding.DestinationType.QUEUE, exchange.getName(), "", null));
                SimpleMessageListenerContainer container = new SimpleMessageListenerContainer();
                container.setConnectionFactory(connectionFactory);
                container.setQueueNames(queue.getName());
                container.setMessageListener(new MessageListenerAdapter(consumer));
            }else{
                //TODO
            }
        }
    }

由于SimpleMessageListenerContainer是生命周期Bean,因此除非在@Configuration Bean中使用,否则它将无法工作

我需要在运行时以编程方式创建订阅,以便Bean从队列中获取事件。 任何想法如何做到这一点?

目前尚不清楚您的问题是什么。 您可以在运行时创建容器。 只需在配置它后调用start() 有很多测试用例可以做到这一点。

您还可以将队列添加到现有容器中(并删除它们); 容器的使用者将被回收以从新的队列列表开始使用。

Gary Russells的回答是正确的,但是我的代码仍然存在一些剩余错误:

  • 如果您创建的队列没有名称,RabbitMQ 创建时 会给它一个随机的唯一名称。 声明队列时必须检索它,然后在声明容器和绑定时使用此名称

为了清楚起见,我在最终版本中添加了修改后的代码块:

    public void subscribe(EventType eventType, AbstractEventConsumer consumer) {
    Assert.notNull(eventType);
    Assert.notNull(eventType.toString());
    Assert.isTrue(!eventType.toString().isEmpty());

    Queue queue=new Queue("", false, true, true);
    Map<String, Exchange> beanMap=context.getBeansOfType(Exchange.class);
    if(beanMap!=null&&!beanMap.isEmpty()){
        Exchange exchange=null;
        boolean found=false;
        for(String key : beanMap.keySet()) {
            exchange=beanMap.get(key);
            if(getExchangeName(eventType.toString()).equals(exchange.getName())){
                found=true;
                break;
            }
        }
        if(found){
            String queueName=amqpAdmin.declareQueue(queue);
            amqpAdmin.declareBinding(new Binding(queueName, Binding.DestinationType.QUEUE, exchange.getName(), "", null));
            SimpleMessageListenerContainer container = new SimpleMessageListenerContainer();
            container.setConnectionFactory(connectionFactory);
            container.setQueueNames(queueName);
            container.setMessageListener(new MessageListenerAdapter(consumer));
            container.start();
        }else{
            //TODO
        }
    }
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM