溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何擴(kuò)展Future Response Servlet

發(fā)布時(shí)間:2021-12-27 16:07:53 來(lái)源:億速云 閱讀:116 作者:iii 欄目:編程語(yǔ)言

本篇內(nèi)容介紹了“如何擴(kuò)展Future Response Servlet”的有關(guān)知識(shí),在實(shí)際案例的操作過(guò)程中,不少人都會(huì)遇到這樣的困境,接下來(lái)就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!

作為Web應(yīng)用程序模型的AJAX的出現(xiàn)使服務(wù)器端的面貌發(fā)生了巨大的變化。用戶對(duì)著Web頁(yè)面填寫表單并單擊提交按鈕轉(zhuǎn)到下一個(gè)鏈接的典型Web使用模式現(xiàn)在正在轉(zhuǎn)變?yōu)楦冗M(jìn)的客戶端JavaScript以及功能更豐富的用戶界面,只要對(duì)表單進(jìn)行操作,比如單擊一個(gè)復(fù)選框、按下一個(gè)鍵或?qū)⑹髽?biāo)移到一個(gè)選項(xiàng)卡上,該用戶界面就會(huì)不斷地與服務(wù)器交互。

考慮一下從客戶端傳輸?shù)椒?wù)器的數(shù)據(jù)量有多大。從可用性的角度來(lái)看,用戶在一個(gè)薄客戶端瀏覽器上獲得了富用戶界面,無(wú)需安裝任何東西。但是,當(dāng)在服務(wù)器端擴(kuò)展這些應(yīng)用程序時(shí)就要付出代價(jià)了。AJAX應(yīng)用程序的典型容量規(guī)劃數(shù)可能會(huì)是標(biāo)準(zhǔn)Web應(yīng)用程序的3到4倍。

有人可能會(huì)問(wèn):這對(duì)WebLogic Server有何影響?每個(gè)發(fā)送給WebLogic的HTTP請(qǐng)求都要使用一個(gè)執(zhí)行線程。根據(jù)AJAX編程的性質(zhì)以及許多短期的請(qǐng)求會(huì)以輪詢的形式不斷發(fā)送的情況,該行為模式可能造成大量客戶端請(qǐng)求不斷沖擊服務(wù)器的局面。多年來(lái),WebLogic都將這一問(wèn)題考慮在內(nèi),并構(gòu)建了一個(gè)相當(dāng)棒的特性,即 Future Response Servlet。該范型構(gòu)建于異步servlet理念的基礎(chǔ)之上。從版本6.1開始,該功能就允許開發(fā)人員提供真正異步的來(lái)自服務(wù)器的通知,而無(wú)需對(duì)事件進(jìn)行客戶端輪訓(xùn)并在服務(wù)器端使用執(zhí)行線程。在9.x之前,BEA還不急于公開該類。

如何在現(xiàn)實(shí)中利用該類呢?我們來(lái)看一個(gè)例子。假定業(yè)務(wù)需求是要構(gòu)建一個(gè)基于Web的應(yīng)用程序,該應(yīng)用程序以近乎實(shí)時(shí)的方式向服務(wù)器發(fā)送數(shù)據(jù)而無(wú)需刷新瀏覽器。這樣的應(yīng)用程序可以向服務(wù)器提交一個(gè)需要花很長(zhǎng)時(shí)間處理的請(qǐng)求,而仍然能夠接收到關(guān)于其狀態(tài)的異步事件并監(jiān)聽事件。從技術(shù)角度來(lái)看,這有許多實(shí)現(xiàn)方法。其中一種方法就是使用一個(gè)與Java Servlet通信的Java Applet來(lái)獲得異步信息。這是一種不錯(cuò)的方法,但是對(duì)于用戶來(lái)說(shuō)有些不太方便,因?yàn)樗麄儽仨毾螺d一個(gè)JVM,還要下載一個(gè)applet到瀏覽器。此外,還必須維護(hù)一個(gè)從客戶端到服務(wù)器的持久性套接字連接,以便接收異步消息。設(shè)想一下,如果有1000個(gè)用戶使用該 applet,那么就有1000個(gè)執(zhí)行線程幾乎是在空等著發(fā)送事件通知到客戶端。當(dāng)然了,還有其它方法,比如從applet或AJAX應(yīng)用程序構(gòu)建輪詢機(jī)制來(lái)定期檢查新數(shù)據(jù)。而如果不經(jīng)常接收到數(shù)據(jù),那么輪詢就顯得無(wú)用了,而且還浪費(fèi)了服務(wù)器資源,占用了執(zhí)行線程。反之,服務(wù)器可以定期輪詢,將事件傳播回客戶端,并維護(hù)套接字線程,而無(wú)需使用持久性執(zhí)行線程。這非常類似于Java NIO的運(yùn)行方式。理想情況下,我們都希望構(gòu)建一個(gè)從服務(wù)器“異步”接收事件通知而無(wú)需在服務(wù)器端使用持久性執(zhí)行線程的應(yīng)用程序,不管它是一個(gè) applet還是一個(gè)基于AJAX的薄Web應(yīng)用程序。

此問(wèn)題的一種解決方案是創(chuàng)建一個(gè)擴(kuò)展Future Response Servlet類的servlet。瀏覽器建立了到Future Response Servlet類的單一連接,并在另一個(gè)線程中將它自身注冊(cè)為一個(gè)監(jiān)聽程序。只要在服務(wù)器端接收到一個(gè)事件,線程就向客戶端通知該事件。服務(wù)器與客戶端保持異步,無(wú)需使用持久性執(zhí)行線程。該模型可擴(kuò)展用于多個(gè)并發(fā)用戶的情況。

本文并不打算介紹如何構(gòu)建AJAX應(yīng)用程序。這方面的文章已經(jīng)有很多了。本文的重點(diǎn)在于討論表示層(比如AJAX、applet或者任何前端應(yīng)用程序)的異步處理的重要性。清單1展示了一個(gè)例子。

  1. import java.io.IOException;  

  2. import java.io.PrintWriter;  

  3. import java.util.Date;  

  4. import java.util.Stack;  

  5. import javax.servlet.ServletException;  

  6. import javax.servlet.http.HttpServletRequest;  

  7. import weblogic.servlet.FutureResponseServlet;  

  8. import weblogic.servlet.FutureServletResponse;  

  9. // An AsynchronousServlet that handles HTTP requests from a 
    "separate" thread and  

  10. // not the execute thread used to invoke this servlet.  

  11.  

  12. public class AsynchronousServerResponseServlet extends 
    FutureResponseServlet {  

  13. private final Notifier notifier;  

  14. public AsynchronousServerResponseServlet() {  

  15. this.notifier = new Notifier();  

  16. this.notifier.start();  

  17. }  

  18. public void service(HttpServletRequest request, FutureServletResponse 
    response)  

  19. throws IOException,ServletException {  

  20. // push this client's request to a buffer and return immediately.  

  21. // asynchronous processing occurs in the run method of the Notifier Thread  

  22.  

  23. notifier.poll(request, response);  

  24. }  

  25. class Notifier extends Thread {  

  26. private static Stack clients = new Stack();  

  27. void poll (HttpServletRequest request, FutureServletResponse response) {  

  28. clients.push(new Client(request, response));  

  29. }  

  30. public void run() {  

  31. while (!clients.empty()) {  

  32. Client client = null;  

  33. try{  

  34. client = (Client) clients.pop();  

  35. PrintWriter pw = client.response.getWriter();  

  36. for(int j = 0; j < 10; j++) {  

  37. pw.println("Time is:" + new Date() + "");  

  38. pw.flush();  

  39. }  

  40. pw.close();  

  41. }  

  42. catch(Throwable t) {  

  43. t.printStackTrace();  

  44. }  

  45. finally {  

  46. try {  

  47. client.response.send();  

  48. }  

  49. catch(IOException ioe) {  

  50. ioe.printStackTrace();  

  51. }  

  52. }  

  53. }  

  54. }  

  55. }  

  56. // inner class that holds o-n to the clients http request and response  

  57.  

  58. class Client {  

  59. private HttpServletRequest request;  

  60. private FutureServletResponse response;  

  61. private Client(HttpServletRequest request, FutureServletResponse 
    response) {  

  62. this.request = request;  

  63. this.response = response;  

  64. }  

可以看出,該例子非常簡(jiǎn)單。AsynchronousServerResponseServlet類擴(kuò)展了Future Response Servlet,并重寫了service方法。只使用一個(gè)線程(即Notifier類)來(lái)處理所有的客戶端連接響應(yīng)。對(duì)于每個(gè)HTTP請(qǐng)求,servlet向Notifier線程注冊(cè)套接字連接,然后返回。異步事件被交付給客戶端,而持久性套接字連接被維持。

單個(gè)線程可管理多個(gè)客戶端連接!run()方法可用于根據(jù)某種消息選擇條件回調(diào)事件到客戶端。該例子只執(zhí)行了一個(gè)服務(wù)器端的push操作,有些過(guò)分簡(jiǎn)單了。線程池可被用于某些類型的事件處理。

“如何擴(kuò)展Future Response Servlet”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí)可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI