最近在开发Spring项目,其中用到了多个RabbitMQ,现在总结如下。
如果要配置多个RabbitMQ,就将rabbitMq相关的xml中的配置再复制一份,然后创建对应的生产者消费者类即可。
1.pom.xml中进行配置,导入jar包:
<dependency>
<groupId>org.springframework.amqp</groupId>
<artifactId>spring-rabbit</artifactId>
<version>1.3.5.RELEASE</version>
</dependency>
<dependency>
<groupId>com.rabbitmq</groupId>
<artifactId>amqp-client</artifactId>
<version>3.3.4</version>
</dependency>
2.创建rabbitmq的xml配置文件,路径例如:resources/context/context-rabbitMq.xml;这个路径与xml文件名可以自定义,只要spring配置文件中把这个xml包含了就行;内容如下:
<!-- 配置rabbitMq链接 -->
<bean id="connectionFactory" class="org.springframework.amqp.rabbit.connection.CachingConnectionFactory">
<property name="username" value="${username}"/>
<property name="password" value="${password}"/>
<property name="port" value="${port}"/>
<property name="virtualHost" value="${vhost}"/>
<property name="channelCacheSize" value="50"/>
<property name="connectionTimeout" value="5000"/>
<property name="addresses" value="${addresses}"/>
</bean>
<!-- 配置admin信息后,当前producer中的exchange和queue会在rabbitmq服务器上自动生成 -->
<rabbit:admin connection-factory="connectionFactory" />
<!-- 消息队列消费者 -->
<bean id="myConsumer" class="com.test.MyConsumer" />
<rabbit:listener-container connection-factory="connectionFactory">
<rabbit:listener ref="myConsumer" queue-names="${queue}" />
</rabbit:listener-container>
<!-- rabbitMQ实例,生产者用 -->
<bean id="rabbitTemplate" class="org.springframework.amqp.rabbit.core.RabbitTemplate">
<property name="connectionFactory" ref="connectionFactory" />
<property name="channelTransacted" value="true" />
</bean>
<!-- 如果有多个mq,再复制一次上面的配置即可,bean的id设置成不同的,可以使用不同的链接信息 -->
3.编写properties配置文件,路径如:resources/properties/config.properties;路径与文件名可自定义,spring配置文件中把这个文件包含了就行,如:
<bean name="propertyConfiger" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer" >
<property name="ignoreResourceNotFound" value="true" />
<property name="locations">
<list>
<value>classpath:properties/config.properties</value>
</list>
</property>
</bean>
“classpath:”指的就是resources文件夹。
config.properties内容如下,与rabbitmq中的${}对应:
host=10.111.11.111,10.111.11.112,10.111.11.113
port=5672
username=admin
password=admin
#交换机名,mq生产者用
exchange=e1.exchange
queue=q1.queue
vhost=v1.vhost
4.编写消费者类MyConsumer.java,这个类用来监听mq队列消息,如果队列中有数据,则拿出一个来处理。如下:
public class MyConsumer implements MessageListener {
private static final Logger log = LoggerFactory.getLogger(MyConsumer.class);
@Override
public void onMessage(Message message){
String msg = "";
MyMsgBean bean = null;
try{
//这里有个小坑,下面会提到
//msg = new String(message.getBody(), message.getMessageProperties().getContentEncoding());
//获取mq中的消息,从byte[]转为string
msg = new String(message.getBody());
//项目中,这个msg是json格式的String,再把String转为bean,容易处理些
bean = JSONObject.parseObject(msg, MyMsgBean.class);
//TODO 以下省略处理步骤
}
catch(Exception e){
log.error("mq消费者异常",e);
}
}
}
这个类在上方的context-rabbitMq.xml中配置过了(myConsumer),所以能实现自动监听mq队列并处理的功能。
5.编写生产者类MyProducer.java,这个类通过使用rabbitTemplate对象,把信息放入mq队列。如下:
public class MyProducer{
private static final Logger log = LoggerFactory.geetLogger(MyProducer.class);
//交换机名称
@Value("${e1.exchange}")
private String exchange;
//@Autowired
private RabbitTemplate rabbitTemplate;
public void setRabbitTemplate(RabbitTemplate rabbitTemplate){
this.rabbitTemplate = rabbitTemplate;
}
public RabbitTemplate getRabbitTemplate(){
return this.rabbitTemplate;
}
public void sendMsg(String msg){
Message info = MessageBuilder.withBody(msg.getBytes())
.setContentType(MessageProperties.CONTENT_TYPE_JSON)
.setContentEncoding("utf-8")
.build();
try{
rabbitTemplate.send(exchange,"",info);
}catch(Exception e){
log.error("生产者将信息放入mq失败!",e);
}
}
}
这个类中的rabbitTemplate是第一步在xml中配置好的,所以能注入,用set方法注入的(也可以用注解注入,还可以用构造方法注入);
这个类本身也是在xml中配置的,然后其它类就可以注入这个类,调用这个类的sendMsg()方法把消息存入mq队列了(例如Controller.java类中注入这个类,然后给mq新增消息)
配置如:
<bean id="myProducer" class="com.test.MyProducer">
<property name="rabbitTemplate" ref="rabbitTemplate"></property>
</bean>
6.其它:
直接输入mq的ip,然后输入mq的账号密码,可以登录mq管理中心;
在这个界面,点击queues,找到自己在使用的队列,可以手动给队列添加信息;
具体如下:
(1)找到Publish message下拉框;
(2)Headers中可以写key与value,对应Message对象中的Headers字段。
(3)Properties中可以写key与value,对应Message对象中的某些字段,点击旁边的"?"查看有效果的key,区分大小写;例如:
content_encoding=utf-8
(4)Properties中可以写具体信息,对应Message对象的getBody()方法获得的内容,例如:
{"json":"123"}