中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

究 Spring 3.1之無web.xml式 基于代碼配置的servlet3.0應用

發布時間:2020-05-20 14:50:41 來源:網絡 閱讀:543 作者:palmtale 欄目:web開發
       (同樣寫于11年12月20日左右的,轉導入此)

        大家應該都已經知道Spring 3.1對無web.xml式基于代碼配置的servlet3.0應用。通過spring的api或是網絡上高手們的博文,也一定很快就學會并且加到自己的應用中去了。PS:如果還沒,也可以小小參考一下鄙人的上一篇文章<<探 Spring 3.1之無web.xml式 基于代碼配置的servlet3.0應用>>。 

      經過一天的深度research, 我了解,理解以及重現了springframework的那一小段代碼。
 

        OK,第一步,入手點,WebApplicationInitializer接口。因為我們只需實現這個接口覆寫它的一個方法,就可以做到配置web.xml同樣的功效。看它的源碼,其實看和不看沒什么兩樣: 

Java代碼 
  1. package org.springframework.web;  
  2.   
  3. import javax.servlet.ServletContext;  
  4. import javax.servlet.ServletException;  
  5. public interface WebApplicationInitializer {  
  6.     void onStartup(ServletContext servletContext) throws ServletException;  
  7. }  


       就這么點兒,有效代碼5行,弄地我一頭霧水,就是一個普通接口,聲明了一個方法。連注解都沒有,server是怎么找到實現了它的類的?如果這樣,何不找我定義的其它接口(的實現類完成配置工作)呢。可見現在java的解耦技術,真令人汗顏。 


   第二步,這個接口旁邊(同包)有個SpringServletContainerInitializer, 看下它是何方神圣吧:
 

Java代碼 
  1. package org.springframework.web;  
  2.   
  3. import java.lang.reflect.Modifier;  
  4. import java.util.Collections;  
  5. import java.util.LinkedList;  
  6. import java.util.List;  
  7. import java.util.ServiceLoader;  
  8. import java.util.Set;  
  9. import javax.servlet.ServletContainerInitializer;  
  10. import javax.servlet.ServletContext;  
  11. import javax.servlet.ServletException;  
  12. import javax.servlet.annotation.HandlesTypes;  
  13.   
  14. import org.springframework.core.annotation.AnnotationAwareOrderComparator;  
  15.   
  16. @HandlesTypes(WebApplicationInitializer.class)  
  17. public class SpringServletContainerInitializer implements ServletContainerInitializer {  
  18. public void onStartup(Set<Class<?>> webAppInitializerClasses, ServletContext servletContext)  
  19.             throws ServletException {  
  20.   
  21.         List<WebApplicationInitializer> initializers = new LinkedList<WebApplicationInitializer>();  
  22.         if (webAppInitializerClasses != null) {  
  23.             for (Class<?> waiClass : webAppInitializerClasses) {  
  24.                 // Be defensive: Some servlet containers provide us with invalid classes,  
  25.                 // no matter what @HandlesTypes says...  
  26.                 if (!waiClass.isInterface() && !Modifier.isAbstract(waiClass.getModifiers()) &&             WebApplicationInitializer.class.isAssignableFrom(waiClass)) {  
  27.                     try {  
  28.                         initializers.add((WebApplicationInitializer) waiClass.newInstance());  
  29.                     }  
  30.                     catch (Throwable ex) {  
  31.                         throw new ServletException("Failed to instantiate WebApplicationInitializer class", ex);  
  32.                     }  
  33.                 }  
  34.             }  
  35.         }  
  36.   
  37.         if (initializers.isEmpty()) {  
  38.             servletContext.log("No Spring WebApplicationInitializer types detected on classpath");  
  39.             return;  
  40.         }  
  41.   
  42.         Collections.sort(initializers, new AnnotationAwareOrderComparator());  
  43.         servletContext.log("Spring WebApplicationInitializers detected on classpath: " + initializers);  
  44.   
  45.         for (WebApplicationInitializer initializer : initializers) {  
  46.             initializer.onStartup(servletContext);  
  47.         }  
  48.     }  
  49.   
  50. }  



     以上的有效代碼28行。剛看時也很迷茫,其實慢慢就理解了。擬個偽代碼吧,方便大家理解: 
      1,定義一個類SpringServletContainerInitializer,并標明該類要操作的一個類WebApplicationInitializer 
      2, 該類會行使ServletContainerInitializer接口的一個行為onStartup,從而將一個集合中的初始化設置 全部配置到ServletContext的實例中。 
      3,具體的onStartup方法中,建立合格配置列表, 
      4,如果確定集合中有配置,逐一檢查配置是否是合格配置,具體判斷依據:這個類不是接口,不是抽象類,而且是所要操作的那個接口的一個實現類。滿足此依據,合格。將合格的配置類實例化放入合格配置列表。過程中有錯要通知控制臺。 
     5,如若執行完步驟4,發現沒有合格配置,在ServletContext記錄該結果,并結束onStartup行為。 
     6,將找到配置按一定排列方式(AnnotationAwareOrder)排序。 
     7,在ServletContext中記錄找到結果。 
     8,逐一執行配置。 即驅動每一個WebApplicationInitializer的實現類行使其onStartup行為。 

     第三步很明顯了,去research 接口ServletContainerInitializer和注解HandleType。在這里:
http://docs.oracle.com/javaee/6/api/javax/servlet/ServletContainerInitializer.html 

    該接口允許一個庫或運行時,(運行時應該指server)聲明為一個web程序的啟動狀態,并執行任何所需的程序中注冊的servlet,filter,listener來響應它...... 
     其它也就不用看了,可以想象得到支持Servlet3機制的服務器,會找到這樣接口的實現類,執行onStartup行為。至于如何找,無非也是這樣一系列的反射機制的應用。自己做一個試試吧: 
     自定義的WebApplicationInitializer:
 

Java代碼 
  1. package com.xxxxx.p_w_picpathcapture.cfg;  
  2.   
  3. import javax.servlet.ServletContext;  
  4. import javax.servlet.ServletException;  
  5.   
  6. public interface WebParameter {  
  7.     public void loadInfo(ServletContext servletContext) throws ServletException;  
  8. }  



     自定義的ServletContainerInitializer,我做得很簡單,直接去執行找到配置類中的loadInfo方法 

Java代碼 
  1. package com.xxxxx.p_w_picpathcapture.cfg;  
  2.   
  3. import java.lang.reflect.Modifier;  
  4. import java.util.Set;  
  5.   
  6. import javax.servlet.ServletContainerInitializer;  
  7. import javax.servlet.ServletContext;  
  8. import javax.servlet.ServletException;  
  9. import javax.servlet.annotation.HandlesTypes;  
  10.   
  11. @HandlesTypes(WebParameter.class)  
  12. public class WebConfiguration implements ServletContainerInitializer {  
  13.   
  14.     @Override  
  15.     public void onStartup(Set<Class<?>> webParams, ServletContext servletCtx)  
  16.             throws ServletException {  
  17.         if (webParams != null) {  
  18.             for (Class<?> paramClass : webParams) {  
  19.                 if (!paramClass.isInterface() && !Modifier.isAbstract(paramClass.getModifiers()) &&  
  20.                         WebParameter.class.isAssignableFrom(paramClass)) {  
  21.                     try {  
  22.                         ((WebParameter) paramClass.newInstance()).loadInfo(servletCtx);  
  23.                     }  
  24.                     catch (Throwable ex) {  
  25.                         throw new ServletException("Failed to instantiate WebParam class", ex);  
  26.                     }  
  27.                 }  
  28.             }//loop  
  29.         }//Web Params  
  30.     }//onStartup  
  31.   
  32. }  


      寫個測試Servlet: 

Java代碼 
  1. package com.xxxxx.p_w_picpathcapture.ctrl;  
  2.   
  3. import java.io.IOException;  
  4.   
  5. import javax.servlet.ServletException;  
  6. import javax.servlet.http.HttpServlet;  
  7. import javax.servlet.http.HttpServletRequest;  
  8. import javax.servlet.http.HttpServletResponse;  
  9.   
  10. import com.gxino.p_w_picpathcapture.cfg.WebParameter;  
  11.   
  12. public class TestServlet extends HttpServlet {  
  13.       
  14.     public void doGet(HttpServletRequest req, HttpServletResponse resp){  
  15.         System.out.println("Some client access once");  
  16.         try {  
  17.             req.getRequestDispatcher("/index.jsp").forward(req, resp);  
  18.         } catch (ServletException | IOException e) {  
  19.             // TODO Auto-generated catch block  
  20.             e.printStackTrace();  
  21.         }  
  22.     }  
  23.       
  24. }  



       實現WebParam配置接口來配置剛才的Servlet: 

Java代碼 
  1. package com.xxxxx.p_w_picpathcapture.cfg;  
  2.   
  3. import javax.servlet.ServletContext;  
  4. import javax.servlet.ServletException;  
  5. import javax.servlet.ServletRegistration;  
  6.   
  7.   
  8. public class ServletParameter implements WebParameter {  
  9.   
  10.     @Override  
  11.     public void loadInfo(ServletContext servletContext) throws ServletException {  
  12.         ServletRegistration.Dynamic testServlet=servletContext.addServlet("test","com.gxino.p_w_picpathcapture.ctrl.TestServlet");  
  13.         testServlet.setLoadOnStartup(1);  
  14.         testServlet.addMapping("/index.html");  
  15.     }  
  16.   
  17. }  


        啟動服務器,訪問http://localhost:xxxx/xxxxx/index.html 
  
     失敗。Debug. 發現沒有走這些代碼。應該還差關鍵環節。看來還得知道Servlet3中是怎么找ServletContainerInitializer的。再回剛才ServletContainerInitializer的api有這樣一句:該接口的實現必須聲明一個JAR資源放到程序中的META-INF/services下,并且記有該接口那個實現類的全路徑,才會被運行時(server)的查找機制或是其它特定機制找到。那篇api需要仔細閱讀啊。 


     到org.springframework.web-3.0.1.RELEASE.jar中能找到META-INF/services下的javax.servlet.ServletContainerInitializer文件,內容為org.springframework.web.SpringServletContainerInitializer同樣,我們專門作這樣一個包,在mkdir好的META-INF/services下vi 一個文件命名為javax.servlet.ServletContainerInitializer,內容為自定的那個WebConfiguration的全路徑類名。 然后在META-INF的parent路徑下運行jar cvf test.jar META-INF。一切完畢,將其放到WEB-INF/lib下。啟動。
 
     
     這回大功告成。 
     
     訪問http://localhost:xxxx/xxxxx/index.html。頁面跳到了index.jsp下。 
     并且控制臺打出: Some client access once 

     再使個勁,將Servlet和Servlet配置合二為一:
 

Java代碼 
  1. package com.xxxxx.p_w_picpathcapture.ctrl;  
  2.   
  3. import java.io.IOException;  
  4.   
  5. import javax.servlet.ServletContext;  
  6. import javax.servlet.ServletException;  
  7. import javax.servlet.ServletRegistration;  
  8. import javax.servlet.http.HttpServlet;  
  9. import javax.servlet.http.HttpServletRequest;  
  10. import javax.servlet.http.HttpServletResponse;  
  11.   
  12. import com.xxxxx.p_w_picpathcapture.cfg.WebParameter;  
  13.   
  14. public class TestServlet extends HttpServlet implements WebParameter{  
  15.   
  16.     @Override  
  17.     public void loadInfo(ServletContext servletContext) throws ServletException {  
  18.         ServletRegistration.Dynamic testServlet=servletContext.addServlet("test""com.gxino.p_w_picpathcapture.ctrl.TestServlet");  
  19.         testServlet.setLoadOnStartup(1);  
  20.         testServlet.addMapping("/index.html");  
  21.     }  
  22.     public void doGet(HttpServletRequest req, HttpServletResponse resp){  
  23.         System.out.println("Some client access once");  
  24.         try {  
  25.             req.getRequestDispatcher("/index.jsp").forward(req, resp);  
  26.         } catch (ServletException | IOException e) {  
  27.             // TODO Auto-generated catch block  
  28.             e.printStackTrace();  
  29.         }  
  30.     }  
  31.       
  32. }  



這回我們看到,配置文件與servlet放到了一起。這樣將回節省大量時間。 

    以后直接運用Spring Framework的WebApplicationInitializer也知道是怎么一回事兒了。而且可以將Spring 的applicationContext.xml與web.xml融合在一個類中。即注解為@Configuration,并實現WebApplicationInitializer.回頭試試。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

三都| 靖宇县| 万州区| 星座| 习水县| 西华县| 赣榆县| 铁岭县| 阿尔山市| 环江| 鄂尔多斯市| 丰城市| 仁怀市| 搜索| 铁岭县| 南阳市| 香格里拉县| 通河县| 溆浦县| 西安市| 三原县| 咸阳市| 和平县| 老河口市| 武宁县| 龙州县| 江津市| 庆元县| 东乡| 儋州市| 禄劝| 茂名市| 建平县| 黄平县| 三亚市| 乃东县| 庆云县| 德钦县| 巴彦县| 马山县| 思茅市|