本文转自http://hitmit1314.iteye.com/blog/1315816

大家应该都已经知道Spring 3.1对无web.xml式基于代码配置的servlet3.0应用。通过spring的api或是网络上高手们的博文,也一定很快就学会并且加到自己的应用中去了。PS:如果还没,也可以小小参考一下鄙人的上一篇文章<<探 Spring 3.1之无web.xml式 基于代码配置的servlet3.0应用>>。

经过一天的深度research, 我了解,理解以及重现了springframework的那一小段代码。

OK,第一步,入手点,WebApplicationInitializer接口。因为我们只需实现这个接口覆写它的一个方法,就可以做到配置web.xml同样的功效。看它的源码,其实看和不看没什么两样:

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

就这么点儿,有效代码5行,弄地我一头雾水,就是一个普通接口,声明了一个方法。连注解都没有,server是怎么找到实现了它的类的?如果这样,何不找我定义的其它接口(的实现类完成配置工作)呢。可见现在java的解耦技术,真令人汗颜。

第二步,这个接口旁边(同包)有个SpringServletContainerInitializer, 看下它是何方神圣吧:

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

以上的有效代码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:

  1. package com.gxino.imagecapture.cfg;
  2. import javax.servlet.ServletContext;
  3. import javax.servlet.ServletException;
  4. public interface WebParameter {
  5. public void loadInfo(ServletContext servletContext) throws ServletException;
  6. }

自定义的ServletContainerInitializer,我做得很简单,直接去执行找到配置类中的loadInfo方法

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

写个测试Servlet:

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

实现WebParam配置接口来配置刚才的Servlet:

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

启动服务器,访问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配置合二为一:

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

这回我们看到,配置文件与servlet放到了一起。这样将回节省大量时间。

以后直接运用Spring Framework的WebApplicationInitializer也知道是怎么一回事儿了。而且可以将Spring 的applicationContext.xml与web.xml融合在一个类中。即注解为@Configuration,并实现WebApplicationInitializer.回头试试。

WebApplicationInitializer究 Spring 3.1之无web.xml式 基于代码配置的servlet3.0应用的更多相关文章

  1. servlet3.0无web.xml

    大家应该都已经知道spring 3.1对无web.xml式基于代码配置的servlet3.0应用.通过spring的api或是网络上高手们的博文,也一定很快就学会并且加到自己的应用中去了.PS:如果还 ...

  2. DispatcherServlet和ContextLoaderListener,还有spring+servlet3.0 无web.xml启动问题

    上篇提到: 关于spring +springmvc中两个spring应用上下文(DispatcherServlet和ContextLoaderListener)的问题,挺让人迷糊的. 他们都是加载Be ...

  3. SpringMVC基于代码的配置方式(零配置,无web.xml)直接继承WebMvcConfigurerAdapter

    基于配置文件的web项目维护起来可能会更方便,但是有时候我们会有一些特殊的需求,比如防止客户胡乱更改配置,这时候我们需要给配置隐藏到代码中. 1.创建一个动态web项目(无需web.xml) 2.右键 ...

  4. SpringMVC基于代码的配置方式(零配置,无web.xml)

    基于配置文件的web项目维护起来可能会更方便,可是有时候我们会有一些特殊的需求,比方防止客户胡乱更改配置,这时候我们须要给配置隐藏到代码中. 1.创建一个动态web项目(无需web.xml) 2.右键 ...

  5. web.xml中servlet-mapping的配置

    <servlet-mapping>元素在Servlet和URL样式之间定义一个映射.它包含了两个子元素<servlet- name>和<url-pattern>,& ...

  6. web.xml servlet、servlet-mapping配置

    Servlet常称为服务器端小程序,即运行在服务器端的程序,用于处理及响应客户的请求. Servlet类是个特殊的java类,继承于HttpServlet. --------------------- ...

  7. springmvc学习指南 之---第27篇 spring如何实现servlet3.0无web.xml 配置servlet对象的

    writedby 张艳涛 基于web.xml配置,有人说麻烦,tomcat给按照servlet3.0,实现了基于注解@WebServlet,有人说springmvc的springmvc.xml配置麻烦 ...

  8. Spring之WebContext不使用web.xml启动 初始化重要的类源码分析(Servlet3.0以上的)

    入口: org.springframework.web.SpringServletContainerInitializer implements ServletContainerInitializer ...

  9. Spring配置文件和SpringMVC配置文件 web.xml配置文件 保存自用

    话不多说,最近在周末自己抽时间写一些框架做的系统,当所有东西都需要自己配置时候发现自己压根记不住这么多类和路径,所以日常总结就变得尤为重要了 db-config.properties 将配置文件常量提 ...

随机推荐

  1. CSS--浏览器CSS Hack 收集

    所谓的Hack就是只有特定浏览器才能识别这段hack代码.Hack 不是什么好东西,除非没有办法,我们尽量还是不要用着玩意. 下面是各个浏览器的CSS Hack 列表. Firefox 浏览器 @-m ...

  2. [Spring] Spring Boot 生态

  3. Arbitrage(floyd)

    http://poj.org/problem?id=2240 #include <stdio.h> #include <string.h> <<; ][]; ][] ...

  4. thinkphp data方法

    data方法也是模型类的连贯操作方法之一,用于设置当前要操作的数据对象的值,可能大家不太习惯用这个方法,今天来讲解下如何用好data方法. 用法 写操作 通常情况下我们都是通过create方法或者赋值 ...

  5. go之切片

    一.概念 关于切片 1.切片是对数组一个连续片段的引用,所以切片是一个引用类型 2.切片是数组一样可以索引,可以通过len函数获取切片的数据长度.(数组也可以通过len获取) 3.切片是一个长度可变的 ...

  6. Redis(三)-Ubuntu下安装

    Ubuntu 下安装 在 Ubuntu 系统安装 Redi 可以使用以下命令: $sudo apt-get update $sudo apt-get install redis-server 启动 R ...

  7. 使用javac编译java文件

    过程中遇到的几个问题记录如下: 1.java -version正常显示java版本,但是javac却显示[不是内部外部命令] 原因:JAVA_HOME设置成了用户环境变量,Path里用%JAVA_HO ...

  8. C#将文件压缩成一个文件流,供前端下载

    直接上代码供大家参考... 前端页面就是一个下载的Button.. <body> <form id="form1" runat="server" ...

  9. Visual Studio切换界面显示语言

    [工具]-[选项]-[环境]-[区域设置]-[语言]-[获取其他语言] 安装后重启即可.

  10. Android Retrofit 2.0文件上传

    Android Retrofit 实现(图文上传)文字(参数)和多张图片一起上传 使用Retrofit进行文件上传,肯定离不开Part & PartMap. public interface ...