A "Hello" program on EJB3 with Jboss server 4.2.2GA (PART 2) - EJB3 MDB with Jboss server.

              MDB is the MVP(most valuable player) both in previous EJB version and EJB3. Although coding with MDB is simple in EJB2.x, EJB3 make it much more friendly to you. Let's get to the point.
              Following is what you should know before write the first MDB in EJB3:
          •     MDB is just a POJO which must implement javax.jms.MessageListener directly or indirectly.
          •     MDB can not extends from another MDB. This should be fine, because each MDB should has its own destination or queue.
          •     Do not throw runtime exception which will cause the instance of this MDB to be destroyed.
          •     One no argument construct method like session bean does.
              Let's see how everything goes on, first of all is the new version EJB3 MDB example source code:
             
           1 package com.ramon.expejb3.session.impl;
           2 
           3 import javax.annotation.PostConstruct;
           4 import javax.annotation.PreDestroy;
           5 import javax.annotation.Resource;
           6 import javax.ejb.ActivationConfigProperty;
           7 import javax.ejb.MessageDriven;
           8 import javax.jms.JMSException;
           9 import javax.jms.Message;
          10 import javax.jms.MessageListener;
          11 import javax.jms.Queue;
          12 import javax.jms.QueueConnection;
          13 import javax.jms.QueueConnectionFactory;
          14 import javax.jms.QueueSender;
          15 import javax.jms.QueueSession;
          16 import javax.jms.Session;
          17 import javax.jms.TextMessage;
          18 
          19 @MessageDriven(
          20         name = "greetingSender",
          21         activationConfig = {
          22                 @ActivationConfigProperty(propertyName="destinationType", propertyValue="javax.jms.Queue"),
          23                 @ActivationConfigProperty(propertyName="destination", propertyValue="queue/ramonQueue")
          24         }
          25 )
          26 public class GreetingSenderMDB implements MessageListener {
          27     
          28     @Resource(mappedName="java:/XAConnectionFactory")
          29     private QueueConnectionFactory qconFactory;
          30     
          31     @Resource(mappedName="queue/ramonRecoderQueue")
          32     private Queue queue;
          33     
          34     private QueueConnection qcon;
          35     
          36     private QueueSession qsession;
          37     
          38     private QueueSender qsender;
          39     
          40     private TextMessage msg;
          41     
          42     @PostConstruct
          43     public void init() {
          44         try {
          45             qcon = qconFactory.createQueueConnection();
          46             qsession = qcon.createQueueSession(false, Session.AUTO_ACKNOWLEDGE);
          47             qsender = qsession.createSender(queue);
          48             msg = qsession.createTextMessage();
          49             qcon.start();
          50             System.out.println(this.getClass() + " init done.");
          51         } catch (JMSException e) {
          52             // TODO Auto-generated catch block
          53             e.printStackTrace();
          54         }
          55     }
          56     
          57     private void send(String message) throws JMSException {
          58         msg.setText(message);
          59         qsender.send(msg);
          60     }
          61     
          62     public void onMessage(Message arg0) {
          63         TextMessage txt = (TextMessage)arg0;
          64         try {
          65             System.out.println("Message '" + txt.getText() + "' has been received.");
          66             send(txt.getText());
          67             System.out.println(">>> Record msg for '" + txt.getText() + "' has been sent out.");
          68         } catch (JMSException e) {
          69             // TODO Auto-generated catch block
          70             e.printStackTrace();
          71         }
          72     }
          73     
          74     @PreDestroy
          75     public void gc() {
          76         try {
          77             qcon.close();
          78             System.out.println("GC for " + this.getClass() + ".");
          79         } catch (JMSException e) {
          80             // TODO Auto-generated catch block
          81             e.printStackTrace();
          82         }
          83     }
          84 
          85 }
          86 


              This is a smiple example, we use @MessageDriven to make this POJO a EJB3 MDB, the "destinationType" attribute make this MDB register itself to a JMS queue not a topic, the "destination" attribute tell the MDB where to listen for the queue, it's a jndi name of the queue you specified in your container. Other part of this code is also self-explanation, you should just focus on how to implement the MessageListener interface -- the onMessage() method.

              For our example, the logic in onMessage() is simple, described as follow:
              1. Receive message from client invocation.
              2. Create a new message according to received message and then send it out to another JMS queue.
              Let's see the source code:
             
           1    public void onMessage(Message arg0) {
           2         TextMessage txt = (TextMessage)arg0;
           3         try {
           4             System.out.println("Message '" + txt.getText() + "' has been received.");
           5             send(txt.getText());
           6             System.out.println(">>> Record msg for '" + txt.getText() + "' has been sent out.");
           7         } catch (JMSException e) {
           8             // TODO Auto-generated catch block
           9             e.printStackTrace();
          10         }
          11     }
              Yup, it's simple, just like the ordinary JMS listener implementation. Let's see how does the send() method in the line 5 get the ConnectionFactory and Queue object and then send message to another queue. EJB3 give us an anotation named "Resource", with this anotation container can inject resource such as DataSourceConnectionFactory, JMSConnectionFactory... into our bean instance, we use this anotation to inject JMSConnectionFactory and JMSQueue, see the code snatch:
             
          1     @Resource(mappedName="java:/XAConnectionFactory")
          2     private QueueConnectionFactory qconFactory;
          3     
          4     @Resource(mappedName="queue/ramonRecoderQueue")
          5     private Queue queue;

              With this anotation we can remove the boring JNDI lookup code, that's really great, because I always copy and paste for JNDI lookup code:) What important is that you should use the "mappedName" attribute instead of the "name" attribute when you want to lookup some JNDI, because the "name" attribute always triger an "env" prefix before the actual JNDI name you specified.

              Other part of this code is simple, so I just paste the code here, GreetingRecordMDB.java:
           1 package com.ramon.expejb3.session.impl;
           2 
           3 import javax.ejb.ActivationConfigProperty;
           4 import javax.ejb.MessageDriven;
           5 import javax.jms.JMSException;
           6 import javax.jms.Message;
           7 import javax.jms.MessageListener;
           8 import javax.jms.TextMessage;
           9 
          10 @MessageDriven(
          11         name = "greetingRecoder",
          12         activationConfig = {
          13                 @ActivationConfigProperty(propertyName="destinationType", propertyValue="javax.jms.Queue"),
          14                 @ActivationConfigProperty(propertyName="destination", propertyValue="queue/ramonRecoderQueue")
          15         }
          16 )
          17 public class GreetingRecordMDB implements MessageListener {
          18 
          19     public void onMessage(Message arg0) {
          20         TextMessage msg = (TextMessage)arg0;
          21         try {
          22             String name = msg.getText();
          23             System.out.println(name + "has been recorded.");
          24         } catch (JMSException e) {
          25             // TODO Auto-generated catch block
          26             e.printStackTrace();
          27         }
          28     }
          29 
          30 }
          31 
              The client side code GreetingSenderMDBTest.java:
           1 package com.ramon.expejb3.session.impl;
           2 
           3 import javax.jms.JMSException;
           4 import javax.jms.Queue;
           5 import javax.jms.QueueConnection;
           6 import javax.jms.QueueConnectionFactory;
           7 import javax.jms.QueueSender;
           8 import javax.jms.QueueSession;
           9 import javax.jms.Session;
          10 import javax.jms.TextMessage;
          11 import javax.naming.Context;
          12 
          13 import com.ramon.expejb3.session.ExpEJB3BaseTestCase;
          14 
          15 public class GreetingSenderMDBTest extends ExpEJB3BaseTestCase {
          16 
          17     private QueueConnectionFactory qconFactory;
          18     private QueueConnection qcon;
          19     private QueueSession qsession;
          20     private QueueSender qsender;
          21     private Queue queue;
          22     private TextMessage msg;
          23 
          24     @Override
          25     protected void setUp() throws Exception {
          26         // TODO Auto-generated method stub
          27         super.setUp();
          28         Context ctx = this.getContext();
          29         qconFactory = (QueueConnectionFactory) ctx
          30                 .lookup("java:/XAConnectionFactory");
          31         qcon = qconFactory.createQueueConnection();
          32         qsession = qcon.createQueueSession(false, Session.AUTO_ACKNOWLEDGE);
          33         queue = (Queue) ctx.lookup("queue/ramonQueue");
          34         qsender = qsession.createSender(queue);
          35         msg = qsession.createTextMessage();
          36         qcon.start();
          37     }
          38 
          39     public void send(String message) throws JMSException {
          40         msg.setText(message);
          41         qsender.send(msg);
          42     }
          43 
          44     public void close() throws JMSException {
          45         qcon.close();
          46     }
          47 
          48     @Override
          49     protected void tearDown() throws Exception {
          50         // TODO Auto-generated method stub
          51         super.tearDown();
          52     }
          53 
          54     public void testOnMessage() {
          55         try {
          56             for (int i = 0; i < 5; i++) {
          57                 String msg = "Ramon " + i;
          58                 send(msg);
          59                 System.out.println("send msg: " + msg);
          60             }
          61             send("end");
          62         } catch (JMSException e) {
          63             // TODO Auto-generated catch block
          64             e.printStackTrace();
          65         } finally {
          66             try {
          67                 close();
          68             } catch (JMSException e) {
          69                 // TODO Auto-generated catch block
          70                 e.printStackTrace();
          71             }
          72         }
          73     }
          74 
          75 }
          76 

              The JMS configuration snatch in file "jbossmq-destinations-service.xml" of Jboss server:
           1 <mbean code="org.jboss.mq.server.jmx.Queue"
           2      name="jboss.mq.destination:service=Queue,name=ramonQueue">
           3     <depends optional-attribute-name="DestinationManager">jboss.mq:service=DestinationManager</depends>
           4     <depends optional-attribute-name="SecurityManager">jboss.mq:service=SecurityManager</depends>
           5     <attribute name="MessageCounterHistoryDayLimit">-1</attribute>
           6     <attribute name="SecurityConf">
           7       <security>
           8         <role name="guest" read="true" write="true"/>
           9         <role name="publisher" read="true" write="true" create="false"/>
          10         <role name="noacc" read="false" write="false" create="false"/>
          11       </security>
          12     </attribute>
          13   </mbean>
          14   
          15   <mbean code="org.jboss.mq.server.jmx.Queue"
          16      name="jboss.mq.destination:service=Queue,name=ramonRecoderQueue">
          17     <depends optional-attribute-name="DestinationManager">jboss.mq:service=DestinationManager</depends>
          18     <depends optional-attribute-name="SecurityManager">jboss.mq:service=SecurityManager</depends>
          19     <attribute name="MessageCounterHistoryDayLimit">-1</attribute>
          20     <attribute name="SecurityConf">
          21       <security>
          22         <role name="guest" read="true" write="true"/>
          23         <role name="publisher" read="true" write="true" create="false"/>
          24         <role name="noacc" read="false" write="false" create="false"/>
          25       </security>
          26     </attribute>
          27   </mbean>



          posted on 2008-03-25 17:42 Find it, try it, experience it 閱讀(1126) 評論(0)  編輯  收藏


          只有注冊用戶登錄后才能發表評論。


          網站導航:
           
          <2008年3月>
          2425262728291
          2345678
          9101112131415
          16171819202122
          23242526272829
          303112345

          導航

          統計

          公告

          If there is any question you have, please don't hesitate, let me know ASAP, you can find me at kenees@gmail.com or QQ: 9808873, hope to make friends with you ;)

          常用鏈接

          留言簿(1)

          隨筆檔案

          文章檔案

          搜索

          積分與排名

          最新評論

          閱讀排行榜

          評論排行榜

          主站蜘蛛池模板: 友谊县| 武鸣县| 调兵山市| 五寨县| 绿春县| 卓资县| 麻江县| 英超| 恭城| 丽水市| 交城县| 庄河市| 汉中市| 西畴县| 马关县| 凤台县| 灌南县| 台湾省| 江孜县| 武穴市| 海林市| 武川县| 柘城县| 府谷县| 中阳县| 铜陵市| 双牌县| 柳州市| 巧家县| 库尔勒市| 成武县| 永兴县| 定边县| 凤冈县| 贵阳市| 陆河县| 鄢陵县| 沭阳县| 清苑县| 峨山| 德钦县|