Java EE 6增强的可扩展性:Web分片和可插拔框架
51CTO编辑推荐:Java EE 6平台指南
Java EE 6中另一个重要的Web层技术是JSF 2.0,它是JSF技术的最新版本,JSF 2.0通过Facelets简化了页面和组件的开发,并支持异步JavaScript和XML(通常叫做Ajax),以及注解。
Servlet 3.0中支持的Web分片
Web应用程序开发人员通常在他们的应用程序中使用第三方框架,如Apache Wicket或Spring MVC。为了使用这些框架,开发人员需要在Web应用程序中注册要使用的框架,一般都使用Web.xml文件来保存注册信息,这个文件还包括构成Web应用程序的Web组件部署描述符,这样使得Web.xml变得非常庞大,难以隔离和维护各种描述符。
Web分片是Servlet 3.0中引入的一个新技术,通过模块化部署描述符解决了这个问题,一个Web分片可以看作是Web.xml文件的一个逻辑段,可以存在多个Web分片,每个分片代表一个逻辑段,Web分片集可以组成一个完整的Web.xml文件,Web.xml文件这种逻辑分区让Web框架自己可以在Web容器中完成注册。你在Web应用程序中使用的每个Web框架可以在一个Web分片中定义所有它需要的工具,如Servlet和监听器,不再需要在Web.xml文件中添加信息。
下面是一个注册Servlet和监听器的Web分片示例:
<web-fragment> <servlet> <servlet-name>myFrameworkServlet</servlet-name> <servlet-class>myFramework.myFrameworkServlet</servlet-class> </servlet> <listener> <listener-class>myFramework.myFrameworkListener</listener-class> </listener> </web-fragment>
一个<web-fragment>元素表示一个Web分片,Web分片必须在一个名叫web-fragment.xml的文件中,这个文件可以放在Web应用程序类路径的任何地方,但Web框架通常会将它的Web分片放到META-INF目录下。
在Web.xml中使用<metadata-complete>元素指示Web容器是否要查找Web分片和注解,如果你将<metadata-complete>设为false,或在Web.xml文件中不指定<metadata-complete>元素,那在以后的开发过程中,容器就必须扫描Web分片和注解,以便为Web应用程序构建有效的元数据。在响应中,Web容器搜索框架JAR文件中的Web分片和注解,Web容器然后使用每个Web分片中的配置信息注册应用程序要使用的框架,但如果将<metadata-complete>设为true,部署描述符会为Web应用程序提供所有的配置信息,这种情况下Web容器不会搜索Web分片和注解。
因为Servlet 3.0技术支持Web分片,你可以将Web.xml文件模块化,你的Web应用程序仍然有传统的,庞大的Web.xml文件,但它是一个包含一到多个Web分片的逻辑分区Web.xml文件。
因为Servlet 3.0让你可以模块化你的部署描述符,这些描述符的处理顺序非常重要,因为它们的顺序影响着调用Servlet,监听器和过滤器的顺序,不过不要担心,因为在Servlet 3.0中你可以指定这些描述符的顺序。Servlet 3.0支持对部署描述符的绝对排序和相对排序,在Web.xml文件中使用<absolute-ordering>元素指定绝对排序,在web-fragment.xml文件中使用<ordering>元素指定相对排序。
例如,假设你的Web应用程序包括两个Web分片:MyFragment2 和 MyFragment3,也包括一个Web.xml文件,可以象下面这样声明描述符使用绝对排序:
<web-app> <name>MyApp</name> <absolute-ordering> <name>MyFragment3</name> <name>MyFragment2</name> </absolute-ordering> ... </web-app>
通过上面的代码,我们可以看出处理顺序如下:
◆web.xml - web.xml描述符总是第一个处理。
◆MyFragment3
◆MyFragment2
可插拔式共享框架
Web分片和注解不是Servlet 3.0允许你扩展web应用程序的唯一方法,你也可以插入构建在Web容器之上的框架共享副本,如基于XML Web服务的Java API(JAX-WS),JAX-RS 和 JSF,Servlet 3.0引入了一个新接口ServletContainerInitializer,它可以用来插入一个框架。
下面是一个插入框架A的例子:
@HandlesTypes(AnnotationA.class) AServletContainerInitializer implements ServletContainerInitializer { public void onStartup(Set<Class<A>>c, ServletContext ctx) throws ServletException { // Framework-specific code here to initialize the runtime // and setup the mapping etc. ServletRegistration reg = ctx.addServlet("AServlet", "com.foo.AServlet"); reg.addServletMapping("/foo");