通过前篇介绍对JMS机制有些了解,应用JMS可以实现两个组件之间的通信,发送消息的一方称为消息生产者,接收消息的一方称为消息消费者。消息生产者可以由任何JavaEE组件来充当,而消息消费者分为同步消费者和异步消费者两种。如果使用同步消费者,消费者从消息目的读不到消息,就会导致当前线程被阻塞,程序无法向下执行。因此,项目中应用大多使用异步消费者。异步消费者需要一直监听消息目的,当JMS消息被送达消息目的时,消息异步消费者被触发,进而读取消息并对消息进行处理。
MDB是一个JMS消息的异步消息消费者。它由无状态的session Bean演化而来,其区别在于:无状态session Bean提供了业务接口,可以被客户端直接调用,与客户端之间是同步的通信方式;MDB只是一个消息监听者,不能被客户端直接调用。
所以,在实际开发中,应该根据需求来选择stateless session bean和MDB。1. 如果该业务方法的处理时间比较长,而且处理时间可能具有一定的不确定性。2. 客户端不许理解得到该方法的返回结构。以上情况时可以使用MDB。
MDB的使用:
MDB作为消息监听器,不用对外提供接口,但是,需要实现MessageListener接口。
下面是一个简单的MDB:
import javax.ejb.ActivationConfigProperty;
import javax.ejb.MessageDriven;
import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.MessageListener;
import javax.jms.TextMessage;
@MessageDriven(
activationConfig = {
//指定MDB所监听消息目的的类型
@ActivationConfigProperty(propertyName = "destinationType", propertyValue = "javax.jms.Queue"),
//指定MDB所监听的消息目的的JNDI绑定名
@ActivationConfigProperty(propertyName="destination",propertyValue="queue/myqueue")
})
public class MyTopicMDBBean implements MessageListener{
//实现onMessage()方法,当JMS消息被送达消息目的时被触发
public void onMessage(Message msg) {
try {
TextMessage message =(TextMessage) msg;
System.out.println(message.getText());
} catch (JMSException e) {
e.printStackTrace();
}
}
}
客户端代码就是介绍JMS时的那几步:
public class MyQueueMDBbeanClient {
private static Context context;
public static void main(String[] args) throws Exception{
Context context = getInitialContext();
//JMS客户端使用JNDI查找、定位连接工厂,
ConnectionFactory factory = (ConnectionFactory)context.lookup("ConnectionFactory");
//消息目的
Destination dest = (Destination)context.lookup("queue/myqueue");
//JMS连接,由连接工厂创建
Connection connection = (Connection)factory.createConnection();
//参数false表示创建的会话是否具有事务性,第二个参数表示消息的确认方式,即当消息接收者接收到消息,并作出了对应处理后,它将回送一个确认消息
Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
//消息生产者
MessageProducer sender = session.createProducer(dest);
sender.setDeliveryMode(DeliveryMode.PERSISTENT);
sender.setTimeToLive(20000);
//通过JMS会话创建一个文本消息
TextMessage msg = session.createTextMessage();
msg.setText("HelloWord");
sender.send(msg);
session.close();
connection.close();
}
protected static Context getInitialContext() throws Exception {
if(context == null){
Hashtable props = new Hashtable();
props.put(Context.INITIAL_CONTEXT_FACTORY, "org.jnp.interfaces.NamingContextFactory");
props.put(Context.URL_PKG_PREFIXES, "org.jboss.naming:org.jnp.interfaces");
props.put(Context.PROVIDER_URL, "jnp://localhost");
context = new InitialContext(props);
}
return context;
}
}
使用服务器是JBoss,客户端代码使用的是JMS1.1规范,可以直接使用ConnectionFactory、Destination、MessageProducer等接口,不用区分使用的是PTP消息模型还是pub-sub模型。