-
Notifications
You must be signed in to change notification settings - Fork 691
Spring supports
本指南以1.4.5版本的java客户端为起点编写。
- 简单例子
- 消息
- 客户端配置
- 会话工厂MessageSessionFactory
- 发送消息MessageProducer
- 订阅消息MessageConsumer
- 遍历消息TopicBrowser
- Spring框架支持
- 高级主题
在阅读本节之前,建议您先阅读前面的章节,系统了解客户端API的使用。从1.4.5开始,MetaQ提供了一套API用来支持Spring框架,基本形式类似spring对JMS的支持。下面的例子只在spring 3.0框架下测试通过,没有测试spring 2.x版本。
在Sring容器内配置一个MessageSessionFactory
:
<!-- message session factory -->
<bean id="sessionFactory" class="com.taobao.metamorphosis.client.extension.spring.MetaqMessageSessionFactoryBean">
<property name="zkConnect" value="127.0.0.1:2181"/>
<property name="zkSessionTimeoutMs" value="30000"/>
<property name="zkConnectionTimeoutMs" value="30000"/>
<property name="zkSyncTimeMs" value="5000"/>
</bean>
主要是zookeeper参数配置,需要跟服务端的zk配置保持一致。更多参数参见AbstractMetaqMessageSessionFactory
的javadoc。
简单地将MetaqMessageSessionFactoryBean
替换为XAMetaqMessageSessionFactoryBean
即可,创建的对象将是XAMessageSessionFactory
。
MetaQ的消息载荷是一个byte数组,你需要将具体的消息内容序列化成byte数组才能发送。MessageBodyConverter
接口用于做这个转换:
/**
* Messge body object converter.
*
* @author dennis<[email protected]>
* @since 1.4.5
* @param <T>
*/
public interface MessageBodyConverter<T> {
/**
* Convert a message object to byte array.
*
* @param body
* @return
* @throws MetaClientException
*/
public byte[] toByteArray(T body) throws MetaClientException;
/**
* Convert a byte array to message object.
*
* @param bs
* @return
* @throws MetaClientException
*/
public T fromByteArray(byte[] bs) throws MetaClientException;
}
MetaQ客户端提供了Java序列化实现的JavaSerializationMessageBodyConverter
给你使用,你也可以自定义自己的消息body转换器,比如采用其他序列化协议,如protobufs,hessian等。
配置一个消息body转换器,比如我们就使用Java序列化:
<!-- message body converter using java serialization. -->
<bean id="messageBodyConverter"
class="com.taobao.metamorphosis.client.extension.spring.JavaSerializationMessageBodyConverter"/>
MetaqTemplate
用于发送MetaQ消息,基本配置:
<!-- template to send messages. -->
<bean id ="metaqTemplate" class="com.taobao.metamorphosis.client.extension.spring.MetaqTemplate">
<property name="messageSessionFactory" ref="sessionFactory"/>
<property name="messageBodyConverter" ref="messageBodyConverter"/>
</bean>
metaqTemplate
用到了上面配置的messageSessionFactory
和messageBodyConverter
,用来创建producer和转换消息体。
配置了metaqTemplate
之后,你将可以在你要发送消息的JavaBean里引用这个对象,并使用它发送消息。
MetaqTemplate
的send方法不直接接收Message
对象,而是MessageBuilder
构建器,用来构建消息:
final String topic = "date";
final SendResult sendResult =
template.send(MessageBuilder.withTopic(topic).withBody(new Date());
上面的例子发送topic为date
的消息,消息体为java.util.Date
对象,MetaqTemplate
将调用messageBodyConverter
将消息体的date对象转换为byte数组,构建message对象再发送。
因为我们使用了JavaSerializationMessageBodyConverter
做消息体转换,因此任何可序列化的JavaBean其实都可以作为消息体发送,使用MessageBuilder.withBody
方法设置即可。如果还是想发送自定义的byte数组,可以使用MessageBuilder.withPayload(byte [])
设置。但是两者不能同时设置。
send(MessageBuilder)
方法同样有重载方法用于异步发送和设置请求超时,具体见javadoc。
MetaqTemplate
会为每个发送的topic创建一个MessageProducer
并发布topic,你也可以共享一个MessageProducer
来发送多个topic的消息,只要在配置metqTempalte
的设置属性shareProducer
为true即可:
<!-- template to send messages. -->
<bean id ="metaqTemplate" class="com.taobao.metamorphosis.client.extension.spring.MetaqTemplate">
<property name="messageSessionFactory" ref="sessionFactory"/>
<property name="messageBodyConverter" ref="messageBodyConverter"/>
<property name="shareProducer" value="true"/>
</bean>
同样,MetaqTemplate
可设置默认topic,当使用共享MessageProducer的时候,可设置这个MessageProducer的默认topic,关于默认topic的作用参见发送消息MessageProducer一节。
<!-- template to send messages. -->
<bean id ="metaqTemplate" class="com.taobao.metamorphosis.client.extension.spring.MetaqTemplate">
<property name="messageSessionFactory" ref="sessionFactory"/>
<property name="messageBodyConverter" ref="messageBodyConverter"/>
<property name="shareProducer" value="true"/>
<property name="defaultTopic" value="*"/>
</bean>
MetaQ同样提供了一套API来支持使用Spring框架订阅消息,你除了需要编写消息处理器之外,几乎不用再写代码。
例如你想订阅topic为date
的消息,首先需要配置该topic:
<!-- topics to be subscribed. -->
<bean id = "dateTopic" class="com.taobao.metamorphosis.client.extension.spring.MetaqTopic">
<!-- consumer group -->
<property name="group" value="testGroup"/>
<!-- topic -->
<property name="topic" value="date"/>
<!-- max buffer size to fetch messages -->
<property name="maxBufferSize" value="16384"/>
</bean>
配置了订阅者的分组,想要订阅的topic以及maxBufferSize大小,更多参数见MetaqTopic
的javadoc。
接下来创建消息处理器,继承DefaultMessageListener
类并实现onReceiveMessages(MetaqMessage msg)方法即可:
package com.taobao.metamorphosis.example.spring;
import com.taobao.metamorphosis.client.extension.spring.DefaultMessageListener;
import com.taobao.metamorphosis.client.extension.spring.MetaqMessage;
import java.util.Date;
/**
* Process date messages listener.
*
* @author dennis
*
*/
public class DateMessageListener extends DefaultMessageListener<Date> {
@Override
public void onReceiveMessages(MetaqMessage<Date> msg) {
Date date = msg.getBody();
System.out.println("receive date message:" + date);
}
}
onReceiveMessages
收到的是封装Message
后的MetaqMessage
对象,它会使用你配置的消息体转换器将byte数组转换成java.util.Date
对象,你可以直接通过getBody
获取上文metaqTemplate
例子中发送的日期对象。当然,如果你还需要原始的Message
对象,可以通过getRawMessage()
方法获取:
Message rawMsg = msg.getRawMessage();
编写好消息处理器之后,需要配置消息处理器:
<!-- message listener -->
<bean id= "messageListener" class="com.taobao.metamorphosis.example.spring.DateMessageListener">
<!-- threads to process these messages. -->
<property name="processThreads" value="10"/>
</bean>
processThreads
设置用于处理消息的线程数,如果为0或者负数表示在抓取线程中处理消息。
有了MetaqTopic
和消息处理器之后,我们就可以使用MessageListenerContainer
来真正地订阅消息了:
<!-- listener container to subscribe topics -->
<bean id ="listenerContainer" class="com.taobao.metamorphosis.client.extension.spring.MessageListenerContainer">
<property name="messageSessionFactory" ref="sessionFactory"/>
<property name="messageBodyConverter" ref="messageBodyConverter"/>
<property name="subscribers">
<map>
<entry key-ref="dateTopic" value-ref="messageListener"/>
</map>
</property>
</bean>
在某些版本的Spring里,可能要求subscribers
的配置是这样:
<entry key="bean:name=testTopic" value-ref="messageListener"/>
注意到listenerContainer
引用到messageSessionFactory
和messageBodyConverter
,用于创建consumer和转换收到的消息体。subscribes
是一个map,指定了订阅的topic以及对应的messageListener。你可以订阅多个topic。
MessageListenerContainer
同时可以配置全局的RejectConsumptionHandler
。
同样,MessageListenerContainer
会为每个topic创建一个MessageConsumer
并自动调用subscribe
和complteSubscribe
方法。但是有时候你可能想共享MessageConsumer
,这可以通过设置shareConsumer
为true来实现,同样你也可以设置共享的MessageConsumer
默认订阅的topic,比如前面的配置也可以修改为:
<!-- listener container to subscribe topics -->
<bean id ="listenerContainer" class="com.taobao.metamorphosis.client.extension.spring.MessageListenerContainer">
<property name="messageSessionFactory" ref="sessionFactory"/>
<property name="messageBodyConverter" ref="messageBodyConverter"/>
<property name="shareConsumer" value="true"/>
<property name="defaultTopic" ref="dateTopic"/>
<property name="defaultMessageListener" ref="messageListener"/>
</bean>
MetaqMessageSessionFactoryBean
,MetaqTemplate
,DefaultMessageListener
和MessageListenerContainer
都能正确地处理资源释放,无需额外代码和配置。