EJB消息驅(qū)動Bean

2018-12-07 18:11 更新

消息驅(qū)動bean是一個類型的企業(yè)bean被調(diào)用的EJB容器時,它接收來自隊列或主題的消息。消息驅(qū)動Bean是一個無狀態(tài)bean用來做任務(wù)的異步。

證明使用消息驅(qū)動Bean,我們會利用EJB持久性做以下工作。


  • 步驟1.創(chuàng)建數(shù)據(jù)庫表(參見EJB的持久性章)。

  • 步驟2.創(chuàng)建表對應(yīng)的實體類(參見EJB的持久性章)。

  • 步驟3.創(chuàng)建數(shù)據(jù)源和持久性單元(參見EJB的持久性章)。

  • 步驟4.創(chuàng)建一個無狀態(tài)EJB具有EntityManager實例(參見EJB的持久性章)。

  • 第5步:更新無狀態(tài)EJB。添加記錄,記錄從數(shù)據(jù)庫通過實體管理器添加方法(指EJB持久性章)

  • 第6步:創(chuàng)建一個在JBoss中默認(rèn)應(yīng)用程序目錄中名為BookQueue隊列。

  • 步驟7:基于控制臺應(yīng)用程序的客戶端會發(fā)送消息到這個隊列中。

  • 第8步:創(chuàng)建一個消息驅(qū)動bean將使用無狀態(tài)的bean堅持客戶數(shù)據(jù)。

  • 第9步的JBoss的EJB容器將調(diào)用上述消息驅(qū)動bean并將它傳遞客戶端將被發(fā)送到郵件中。


創(chuàng)建隊列

創(chuàng)建一個名為jbossmq-destinations-service.xml<JBoss Installation Folder> > server > default > deployfolder


在這里,我們創(chuàng)建了一個名為BookQueue的隊列


jbossmq-destinations-service.xml

<mbean code="org.jboss.mq.server.jmx.Queue"  
   name="jboss.mq.destination:service=Queue,name=BookQueue">  
   <depends optional-attribute-name="DestinationManager">
      jboss.mq:service=DestinationManager
   </depends>  
</mbean>  


當(dāng)您啟動JBoss,你會看到在JBoss中日志中的一個類似的條目

...
10:37:06,167 INFO  [QueueService] Queue[/queue/BookQueue] started, fullSize=200000, pageSize=2000, downCacheSize=2000
...


創(chuàng)建消息驅(qū)動Bean

@MessageDriven(
   name = "BookMessageHandler",
   activationConfig = {
      @ActivationConfigProperty( propertyName = "destinationType", 
                                 propertyValue = "javax.jms.Queue"),
      @ActivationConfigProperty( propertyName = "destination", 
                                 propertyValue ="/queue/BookQueue")
   }
)
public class LibraryMessageBean implements MessageListener {
 
   @Resource
   private MessageDrivenContext mdctx;  
 
   @EJB
   LibraryPersistentBeanRemote libraryBean;
 
   public LibraryMessageBean(){        
   }
 
   public void onMessage(Message message) {
   }
}
  • LibraryMessageBean與@MessageDriven注釋annoatated將其標(biāo)記為消息驅(qū)動bean。

  • 其性質(zhì)被定義為destinationType - 隊列和目的地 - /隊列/ BookQueue。

  • 它實現(xiàn)了MessageListener接口暴露出onMessage方法。

  • 它MessgeDrivenContext的資源。

  • LibraryPersistentBeanRemote無狀態(tài)bean是在這個bean持久性的目的注入。

構(gòu)建EjbComponent項目,并在JBoss部署它。構(gòu)建和部署EJB組件后,我們需要一個客戶端發(fā)送消息到JBoss隊列。


示例應(yīng)用程序

讓我們創(chuàng)建一個測試EJB應(yīng)用程序來測試消息驅(qū)動bean。

步驟描述
1用包com.tutorialspoint.entity下一個名字EjbComponentEJB作為解釋的創(chuàng)建項目-創(chuàng)建應(yīng)用程序一章。您也可以使用EJB創(chuàng)建的項目-創(chuàng)建應(yīng)用程序章這樣本章了解EJB的持久性概念。
2包下com.tutorialspoint.entity創(chuàng)建Book.java作為EJB的持久性創(chuàng)建章
3創(chuàng)建LibraryPersistentBean.javaLibraryPersistentBeanRemote作為EJB的持久性創(chuàng)建章
4創(chuàng)建EjbComponent>EjbComponent>來源> conf文件夾安裝文件夾和persistence.xml中 的jboss-ds.xml文件 。這些文件夾可在文件選項卡中可以看出在Netbeans的作為EJB的持久性創(chuàng)建章
5com.tutorialspoint.messagebean下創(chuàng)建LibraryMessageBean.java,并修改它,如下圖所示。
6在JBoss中創(chuàng)建BookQueue隊列如上所述。
7清理并生成應(yīng)用程序,確保業(yè)務(wù)邏輯正在按要求。
8最后,部署JBoss應(yīng)用服務(wù)器上的jar文件的形式應(yīng)用。如果尚未啟動JBoss應(yīng)用服務(wù)器將自動被啟動。
9現(xiàn)在創(chuàng)建EJB客戶端,以同樣的方式一個基于控制臺的應(yīng)用程序在EJB解釋-創(chuàng)建應(yīng)用程序一章的主題創(chuàng)建客戶機(jī)訪問EJB。修改它,如下圖所示。


EJBComponent(EJB組件)

LibraryMessageBean.java

package com.tuturialspoint.messagebean;
 
import com.tutorialspoint.entity.Book;
import com.tutorialspoint.stateless.LibraryPersistentBeanRemote;
import javax.annotation.Resource;
import javax.ejb.ActivationConfigProperty;
import javax.ejb.EJB;
import javax.ejb.MessageDriven;
import javax.ejb.MessageDrivenContext;
import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.MessageListener;
import javax.jms.ObjectMessage;
 
@MessageDriven(
   name = "BookMessageHandler",
   activationConfig = {
      @ActivationConfigProperty( propertyName = "destinationType", 
                                 propertyValue = "javax.jms.Queue"),
      @ActivationConfigProperty( propertyName = "destination", 
                                 propertyValue ="/queue/BookQueue")
   }
)
public class LibraryMessageBean implements MessageListener {
 
   @Resource
   private MessageDrivenContext mdctx;  
 
   @EJB
   LibraryPersistentBeanRemote libraryBean;
 
   public LibraryMessageBean(){        
   }
 
   public void onMessage(Message message) {
      ObjectMessage objectMessage = null;
      try {
         objectMessage = (ObjectMessage) message;
         Book book = (Book) objectMessage.getObject(); 
         libraryBean.addBook(book);
 
      } catch (JMSException ex) {
         mdctx.setRollbackOnly();
      }       
   }   
}


EJBTester(EJB客戶端)

EJBTester.java

package com.tutorialspoint.test;
   
import com.tutorialspoint.entity.Book;
import com.tutorialspoint.stateless.LibraryPersistentBeanRemote;
import java.io.BufferedReader;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.List;
import java.util.Properties;
import javax.jms.ObjectMessage;
import javax.jms.Queue;
import javax.jms.QueueConnection;
import javax.jms.QueueConnectionFactory;
import javax.jms.QueueSender;
import javax.jms.QueueSession;
import javax.naming.InitialContext;
import javax.naming.NamingException;
 
public class EJBTester {
 
   BufferedReader brConsoleReader = null; 
   Properties props;
   InitialContext ctx;
   {
      props = new Properties();
      try {
         props.load(new FileInputStream("jndi.properties"));
      } catch (IOException ex) {
         ex.printStackTrace();
      }
      try {
         ctx = new InitialContext(props);            
      } catch (NamingException ex) {
         ex.printStackTrace();
      }
      brConsoleReader = 
      new BufferedReader(new InputStreamReader(System.in));
   }
   
   public static void main(String[] args) {
 
      EJBTester ejbTester = new EJBTester();
 
      ejbTester.testMessageBeanEjb();
   }
   
   private void showGUI(){
      System.out.println("**********************");
      System.out.println("Welcome to Book Store");
      System.out.println("**********************");
      System.out.print("Options 
1. Add Book
2. Exit 
Enter Choice: ");
   }
   
   private void testMessageBeanEjb(){
 
      try {
         int choice = 1; 
         Queue queue = (Queue) ctx.lookup("/queue/BookQueue");
         QueueConnectionFactory factory =
         (QueueConnectionFactory) ctx.lookup("ConnectionFactory");
         QueueConnection connection =  factory.createQueueConnection();
         QueueSession session = 
         connection.createQueueSession(false, QueueSession.AUTO_ACKNOWLEDGE);
         QueueSender sender = session.createSender(queue);
 
         while (choice != 2) {
            String bookName;
            showGUI();
            String strChoice = brConsoleReader.readLine();
            choice = Integer.parseInt(strChoice);
            if (choice == 1) {
               System.out.print("Enter book name: ");
               bookName = brConsoleReader.readLine();
               Book book = new Book();
               book.setName(bookName);
               ObjectMessage objectMessage = 
                  session.createObjectMessage(book);
               sender.send(objectMessage); 
            } else if (choice == 2) {
               break;
            }
         }
 
         LibraryPersistentBeanRemote libraryBean = 
         (LibraryPersistentBeanRemote)
         ctx.lookup("LibraryPersistentBean/remote");
 
         List<Book> booksList = libraryBean.getBooks();
 
         System.out.println("Book(s) entered so far: " + booksList.size());
         int i = 0;
         for (Book book:booksList) {
            System.out.println((i+1)+". " + book.getName());
            i++;
         }           
      } catch (Exception e) {
         System.out.println(e.getMessage());
         e.printStackTrace();
      }finally {
         try {
            if(brConsoleReader !=null){
               brConsoleReader.close();
            }
         } catch (IOException ex) {
            System.out.println(ex.getMessage());
         }
      }
   }   
}

EJBTester做以下任務(wù)。

  • 從jndi.properties負(fù)荷特性和初始化InitialContext對象。

  • 在testStatefulEjb()方法,JNDI查找與名行 - “/隊列/ BookQueue”以獲得可用的隊列treference在JBoss中。然后發(fā)送者使用隊列會話創(chuàng)建。

  • 然后用戶顯示庫存儲用戶界面和他(她)被要求輸入選擇/。

  • 如果用戶輸入1,系統(tǒng)要求書籍名稱和發(fā)件人發(fā)送書名排隊。當(dāng)JBoss容器接收隊列此消息,它調(diào)用我們的消息驅(qū)動bean的onMessage方法。我們的消息驅(qū)動bean,然后保存使用狀態(tài)會話bean addBook()方法的書。會話Bean是通過持續(xù)的EntityManager稱這本書在數(shù)據(jù)庫中。

  • 如果用戶輸入2,然后又JNDI查找與名行 - “LibraryStatefulSessionBean /遠(yuǎn)程”,再次獲得遠(yuǎn)程業(yè)務(wù)對象(狀態(tài)EJB)和書上市完成。


運(yùn)行客戶端訪問EJB

在項目資源管理器中找到EJBTester.java。右鍵單擊EJBTester類并選擇運(yùn)行文件 。


驗證Netbeans的控制臺下面的輸出。

run:
**********************
Welcome to Book Store
**********************
Options 
1. Add Book
2. Exit 
Enter Choice: 1
Enter book name: Learn EJB
**********************
Welcome to Book Store
**********************
Options 
1. Add Book
2. Exit 
Enter Choice: 2
Book(s) entered so far: 2
1. learn java
1. learn EJB
BUILD SUCCESSFUL (total time: 15 seconds)
  • 上面顯示輸出狀態(tài),我們的消息驅(qū)動bean接收消息和持久性存儲的書和書是從數(shù)據(jù)庫中檢索。


以上內(nèi)容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號