1

我正在尝试使用 spring amqp 使用rabbitmq,下面是我的配置。

<rabbit:connection-factory id="rabbitConnectionFactory"
    port="${rabbitmq.port}" host="${rabbitmq.host}" />

<rabbit:admin connection-factory="rabbitConnectionFactory" />

<rabbit:queue name="${rabbitmq.import.queue}" />

<rabbit:template id="importAmqpTemplate"
    connection-factory="rabbitConnectionFactory" queue="${rabbitmq.import.queue}" />

<beans:bean id="importExchangeMessageListener"
    class="com.stockopedia.batch.foundation.ImportMessageListener" />

<rabbit:listener-container
    connection-factory="rabbitConnectionFactory" concurrency="5">
    <rabbit:listener queues="${rabbitmq.import.queue}" ref="importMessageListener" />
</rabbit:listener-container>

这是一个简单的消息监听器类,

import org.springframework.amqp.core.Message;
import org.springframework.amqp.core.MessageListener;

public class ImportMessageListener implements MessageListener {

    @Override
    public void onMessage(Message message) {
        System.out.println("consumer output: " + message);
    }

}

这是生产者(即春季批次的 itemWriter),

public class ImportItemWriter<T> implements ItemWriter<T> {

    private AmqpTemplate template;

    public AmqpTemplate getTemplate() {
        return template;
    }

    public void setTemplate(AmqpTemplate template) {
        this.template = template;
    }

    public void write(List<? extends T> items) throws Exception {
        for (T item : items) {
            Object reply = template.convertSendAndReceive(item.toString());
            System.out.println("producer output: " + reply);
        }
    }

}

当我运行我的春季批处理作业时, ImportItemWriter.write 被调用。但是 ImportMessageListener.onMessage 不起作用。它不打印消息。我得到控制台上所有项目的低于输出

producer output: null
producer output: null
producer output: null
producer output: null
producer output: null
producer output: null
producer output: null
4

1 回答 1

5

您的消费者没有发送结果...

@Override
public void onMessage(Message message) {
    System.out.println("consumer output: " + message);
}

改成简单的POJO;容器MessageListenerAdapter将为您处理转换,并发送结果。

@Override
public String handleMessage(String message) {
    System.out.println("consumer output: " + message);
    return "result";
}

编辑:

您还没有设置任何交换或路由到您的队列。如果您想使用默认交换/路由,请使用...

convertSendAndReceive("", queueName, item.toString());

编辑2:

或者,将routingKey模板上的 设置为队列名称,然后您可以使用更简单的方法。

这些...sendAndReceive()方法适用于请求/回复场景,因此需要阻塞。要异步执行此操作,您必须使用其中一种...send()方法,并连接您自己的方法SimpleListenerContainer以接收回复;你将不得不做你自己的相关性。利用

public void convertAndSend(Object message, MessagePostProcessor postProcessor)

并在您的消息后处理器中,设置replyTocorrelationId标头...

message.getMessageProperties().setReplyTo("foo");
message.getMessageProperties().setCorrelationId("bar");

或者,Message自己构建对象(例如,使用MessageBuilder)并使用send方法...

template.send(MessageBuilder.withBody("foo".getBytes())
            .setReplyTo("bar")
            .setCorrelationId("baz".getBytes())
            .build());

每个请求都需要一个唯一的correlationId,以便您可以关联响应。

于 2014-04-20T23:25:12.980 回答