經過Servlet3.0添加對異步的支持實現後臺推送

本例參考:http://blog.csdn.NET/chenxiang0207/article/details/14054681/javascript

http://blog.csdn.net/u010497228/article/details/43387575   html

感謝上面的兩位大佬,轉載只是爲了方便瀏覽。java


我按照上面博文的思路從新走了一遍服務器

項目結構以下圖app



[html]  view plain  copy
  1. /**  
  2.  * AsyncServlet  
  3.  *   
  4.  * 支持異步處理的Servlet  
  5.  * 頁面中隱藏的iframe經過訪問此Servlet來創建HTTP長鏈接  
  6.  * 從然後臺能實時的推送javascript代碼給頁面調用  
  7.  *  
  8.  */  
  9. @WebServlet(urlPatterns = "/Async"asyncSupported = true)  
  10. public class AsyncServlet extends HttpServlet {  
  11.   
  12.     private static final long serialVersionUID = 822178713133426493L;  
  13.     private final static int DEFAULT_TIME_OUT = 10 * 60 * 1000;  
  14.   
  15.     @Override  
  16.     protected void doGet(HttpServletRequest req, HttpServletResponse res) {  
  17.         AsyncContext actx = req.startAsync();  
  18.         actx.setTimeout(DEFAULT_TIME_OUT);  
  19.         actx.addListener(new AsyncListener() {  
  20.   
  21.             @Override  
  22.             public void onComplete(AsyncEvent arg0) throws IOException {  
  23.                 // TODO Auto-generated method stub  
  24.                 ClientComet.getInstance().removeAsyncContext(actx);  
  25.                 System.out.println("AsyncListener-->onComplete");  
  26.             }  
  27.   
  28.             @Override  
  29.             public void onError(AsyncEvent arg0) throws IOException {  
  30.                 // TODO Auto-generated method stub  
  31.                 ClientComet.getInstance().removeAsyncContext(actx);  
  32.                 System.out.println("AsyncListener-->onError");  
  33.             }  
  34.   
  35.             @Override  
  36.             public void onStartAsync(AsyncEvent arg0) throws IOException {  
  37.                 // TODO Auto-generated method stub  
  38.                 System.out.println("AsyncListener-->onStartAsync");  
  39.             }  
  40.   
  41.             @Override  
  42.             public void onTimeout(AsyncEvent arg0) throws IOException {  
  43.                 // TODO Auto-generated method stub  
  44.                 ClientComet.getInstance().removeAsyncContext(actx);  
  45.                 System.out.println("AsyncListener-->onTimeout");  
  46.             }  
  47.   
  48.         });  
  49.         ClientComet.getInstance().addAsyncContext(actx);  
  50.     }  
  51. }  

[java]  view plain  copy
  1. /** 
  2.  * ClientComet 
  3.  *  
  4.  * 管理用戶的AsyncContext(添加、刪除) 
  5.  *  
  6.  * 經過開啓一個線程來不斷地從mesgQueue獲取javascript 並遍歷用戶的AsyncContext來吧javascript推送給每一個用戶 
  7.  * 
  8.  */  
  9. public class ClientComet {  
  10.     private static ClientComet instance;  
  11.     private ConcurrentLinkedQueue<AsyncContext> actxQueue;  
  12.     private LinkedBlockingQueue<Javascript> mesgQueue;  
  13.   
  14.     private ClientComet() {  
  15.         actxQueue = new ConcurrentLinkedQueue<AsyncContext>();  
  16.         mesgQueue = new LinkedBlockingQueue<Javascript>();  
  17.         new ClientCometThread().start();  
  18.     }  
  19.   
  20.     public static ClientComet getInstance() {  
  21.         if (instance == null) {  
  22.             instance = new ClientComet();  
  23.         }  
  24.         return instance;  
  25.     }  
  26.   
  27.     public void addAsyncContext(AsyncContext actx) {  
  28.         actxQueue.add(actx);  
  29.     }  
  30.   
  31.     public void removeAsyncContext(AsyncContext actx) {  
  32.         actxQueue.remove();  
  33.     }  
  34.   
  35.     public void callClient(Javascript javascript) {  
  36.         mesgQueue.add(javascript);  
  37.     }  
  38.   
  39.     protected class ClientCometThread extends Thread {  
  40.   
  41.         @Override  
  42.         public void run() {  
  43.             while (true) {  
  44.                 try {  
  45.                     Javascript javascript = mesgQueue.take();  
  46.                     for (AsyncContext actx : actxQueue) {  
  47.                         PrintWriter writer = actx.getResponse().getWriter();  
  48.                         writer.write(javascript.getScript());  
  49.                         writer.flush();  
  50.                         System.out  
  51.                                 .println("ClientCometThread-->sendJavaScript");  
  52.                     }  
  53.                 } catch (InterruptedException | IOException e) {  
  54.                     // TODO Auto-generated catch block  
  55.                     e.printStackTrace();  
  56.                 }  
  57.   
  58.             }  
  59.         }  
  60.     }  
  61.   
  62. }  

[java]  view plain  copy
  1. /** 
  2.  * Javascript 
  3.  * 
  4.  * 提供將javascript的方法調用補全到一個script標籤中的功能 
  5.  */  
  6. public class Javascript {  
  7.     private String script;  
  8.   
  9.     public Javascript(String func) {  
  10.         script = "<script type='text/javascript'>" + "\n" + "window.parent."  
  11.                 + func + "\n" + "</script>" + "\n";  
  12.     }  
  13.   
  14.     public String getScript() {  
  15.         return script;  
  16.     }  
  17. }  

[java]  view plain  copy
  1. /** 
  2.  * TestServlet 
  3.  *  
  4.  * 訪問此Servlet可以給ClientComet的mesgQueue添加對象 
  5.  * 
  6.  */  
  7. @WebServlet(urlPatterns = "/Test")  
  8. public class TestServlet extends HttpServlet {  
  9.   
  10.     private static final long serialVersionUID = -7817902387051107187L;  
  11.   
  12.     @Override  
  13.     protected void doGet(HttpServletRequest req, HttpServletResponse res) {  
  14.         int times = 5;  
  15.         while (--times >= 0) {  
  16.             try {  
  17.                 Thread.sleep(5000);  
  18.                 ClientComet.getInstance().callClient(  
  19.                         new Javascript("append(" + "\'"  
  20.                                 + new Date().toGMTString() + "\'" + ")"));  
  21.                 System.out.println("TestServlet-->callClient");  
  22.             } catch (InterruptedException e) {  
  23.                 // TODO Auto-generated catch block  
  24.                 e.printStackTrace();  
  25.             }  
  26.         }  
  27.     }  
  28. }  

[html]  view plain  copy
  1. Comet.jsp  
  2.   
  3. <%@ page language="java" contentType="text/html; charset=utf-8"  
  4.     pageEncoding="utf-8"%>  
  5. <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">  
  6. <html>  
  7. <head>  
  8. <meta http-equiv="Content-Type" content="text/html; charset=utf-8">  
  9. <title>Comet Test</title>  
  10. <script type='text/javascript'>  
  11.     function append(str) {  
  12.         var textField = document.getElementById("textField");  
  13.         textField.innerHTML = textField.innerHTML + str + "<br/>";  
  14.     };  
  15. </script>  
  16. </head>  
  17. <body>  
  18.     <!-- textField——用來顯示服務器推送的內容 -->  
  19.     <p id="textField"></p>  
  20.     <!-- cometFrame——隱藏的iframe,用來訪問AsyncServlet,創建長鏈接(注意,這樣作window.onload的函數將失效,或直到此鏈接斷開才執行) -->  
  21.     <iframe id="cometFrame" style="display: none;" src="/CometTest/Async"></iframe>  
  22. </body>  
  23. </html>  


 

開啓Tomcat異步

開啓一個頁面A先訪問http://localhost:8080/CometTest/Comet.jspjsp

再打開一個新的頁面B訪問http://localhost:8080/CometTest/Testasync

此時可以看到A的內容不斷的增長ide


PS:若想讓onload函數不失效能夠把對iframe的src的賦值操做放在onload中,而一開始src爲空函數


項目打包:http://download.csdn.net/detail/u010497228/8415545

相關文章
相關標籤/搜索