动态加载

动态加载是 Servlet 3.0 中的新特性,它可以实现在不重启 Web 应用的情况下加载新的 Web 对象(Servlet、Filter、Listener)。

为了实现动态加载的第一种方式:创建 Web 对象和注册 Web 对象到 ServletContext 中分步进行

ServletContext 接口增加了如下方法,用于动态创建 Web 对象:

<T extends Servlet> T createServlet(java.lang.Class<T> c)  throws ServletException  //
<T extends Filter> T createFilter(java.lang.Class<T> c) throws ServletException //
<T extends java.util.EventListener> T createListener(java.lang.Class<T> c) throws ServletException //

例如,如果MyServlet是一个直接或者间接继承 javax.servlet.Servlet 的类,那么就可以通过 createServlet 的方法初始化它:

Servlet myServlet = servletContext.createServlet(MyServlet.class);  // 这里使用了反射技术 

在创建了 Web 对象之后,可以通过 ServletContext 中的如下方法把它注册到 ServletContext 中

ServletRegistration.Dynamic addServlet(java.lang.String servletName, Servlet servlet) //
FilterRegistration.Dynamic addFilter(java.lang.String filterName, Filter filter) //
<T extends java.util.EventListener> void addListener(T t) //

实现动态加载的第二种方式:创建 Web 对象和注册 Web 对象到 ServletContext 中一步完成

使用 ServletContext 中的如下方法

ServletRegistration.Dynamic addServlet(java.lang.String servletName, java.lang.Class<? extends Servlet> servletClass)
ServletRegistration.Dynamic addServlet(java.lang.String servletName, java.lang.String className) FilterRegistration.Dynamic addFilter(java.lang.String filterName, java.lang.Class<? extends Filter> filterClass)
FilterRegistration.Dynamic addFilter(java.lang.String filterName, java.lang.String className) void addListener(java.lang.Class<? extends java.util.EventListener> listenerClass)
void addListener(java.lang.String className)

要创建或者增加Listener,传递给第一个 addListener 方法的类需要实现以下的一个或者多个接口

ServletContextAttributeListener
ServletRequestListener
ServletRequestAttributeListener
HttpSessionListener
HttpSessionAttributeListener

如果 ServletContext 是用于 ServletContextInitializer 中的 onStartup 方法的参数,那么 Listener 也需要实现 ServletContextListener

动态加载实例

package app14a.servlet;

import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; public class FirstServlet extends HttpServlet { // 一个 Servlet,该 Servlet 没有使用 @WebServlet 注解,也没有使用部署描述符来声明。
private static final long serialVersionUID = 1L; // 而是通过使用 Listener 来动态创建、注册、绑定这个 Servlet 并让其生效的。
private String name; public void setName(String name) {
this.name = name;
} protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.setContentType("text/html");
PrintWriter writer = response.getWriter();
writer.println("<html>");
writer.println("<head>");
writer.println("<title>First Servlet</title>");
writer.println("</head>");
writer.println("<body>");
writer.println(name);
writer.println("</body>");
writer.println("</html>");
} protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request, response);
}
}
package app14a.listener;

import javax.servlet.Servlet;
import javax.servlet.ServletContext;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import javax.servlet.ServletRegistration;
import javax.servlet.annotation.WebListener; import app14a.servlet.FirstServlet; @WebListener // 注解
public class DynRegListener implements ServletContextListener { // 监听器 @Override
public void contextInitialized(ServletContextEvent sce) { // ServletContext 创建时,容器调用该方法
ServletContext servletContext = sce.getServletContext(); // 获取 ServletContext 对象实例
Servlet firstServlet = null;
try {
firstServlet = servletContext.createServlet(FirstServlet.class); // 动态创建 Web 对象
} catch (Exception e) {
e.printStackTrace();
}
if (firstServlet != null && firstServlet instanceof FirstServlet) {
((FirstServlet) firstServlet).setName("Dynamically registered servlet");
}
ServletRegistration.Dynamic dynamic = servletContext.addServlet("firstServlet", firstServlet); // 注册到 ServletContext 中
dynamic.addMapping("/dynamic"); // 绑定路径
} @Override
public void contextDestroyed(ServletContextEvent sce) {
}
}

测试结果,Servlet 成功注册到了 ServletContext 中,且绑定到 /dynamic 路径

Servlet容器加载

Servlet 容器加载器也是 Servlet 3.0 中的新特性,对于框架的开发者来说特别有用。

Servlet 容器初始化主要是通过 javax.servlet.ServletContainerInitializer 这个接口。该接口只有一个方法 onStartup。

Servlet 容器中,这个方法在任何 ServletContext 的 Listener 初始化之前都可能会被调用到。

void onStartup(java.util.Set<java.lang.Class<?>> c, ServletContext ctx) throws ServletException  // 

ServletContainerInitializer 的实现类必须使用 HandleTypes 的注解,以便让加载器能够识别。

package app14a.initializer;

import java.util.Set;
import javax.servlet.ServletContainerInitializer;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletRegistration;
import javax.servlet.annotation.HandlesTypes;
import servlet.UsefulServlet; @HandlesTypes({UsefulServlet.class}) // ServletContainerInitializer 的实现类必须使用该注解
public class MyServletContainerInitializer implements ServletContainerInitializer {
@Override
public void onStartup(Set<Class<?>> classes, ServletContext servletContext) throws ServletException { // 该方法的主要任务就是注册 Web 对象
System.out.println("onStartup");
ServletRegistration registration = servletContext.addServlet("usefulServlet", "servlet.UsefulServlet"); // 创建并注册
registration.addMapping("/useful"); // 绑定访问路径
System.out.println("leaving onStartup");
}
}

插件(即 initializer.jar)中的内容

测试结果,

动态加载及Servlet容器加载的更多相关文章

  1. 动态加载机Servlet容器加载器

    动态加载是Servlet 3.0中的新特性,它可以实现在 不重启Web应用的情况下加载新的Web对象(Servlet. Filter.Listener).Servlet容器加载器也是Servlet 3 ...

  2. 对于Servlet、Servlet容器以及一个Servlet容器-Tomcat

    Servlet.Servlet容器等内容讲解 转载自http://blog.csdn.net/iAm333 对于Servlet.Servlet容器以及一个Servlet容器-Tomcat这些概念讲解的 ...

  3. Servlet、Servlet容器等内容讲解

    转载自http://blog.csdn.net/iAm333 对于Servlet.Servlet容器以及一个Servlet容器-Tomcat这些概念讲解的挺清晰的,转载下 之前在开源中国看到一篇文章& ...

  4. JAVA网络编程基本功之Servlet与Servlet容器

    Servlet与Servlet容器关系 Servlet 比较这两个的区别, 就得先搞清楚Servlet 的含义, Servlet (/ˈsərvlit/ ) 翻译成中文就是小型应用程序或者小服务程序, ...

  5. servlet容器处理请求过程

    下图是关于tomcat服务器接收客户请求并作出响应的图例. tomcat不仅仅只是一个servlet容器,也是一个web服务器,servlet容器在web服务器之内或者说servlet容器托管于web ...

  6. servlet 启动加载配置文件及初始化

    在servlet开发中,会涉及到一些xml数据的读取和一些初始化方法的调用.可以在tomcat启动的时候,加载一个servlet去初始化一些数据. 摘自 http://stone02111.iteye ...

  7. springMVC容器加载源码分析

    springmvc是一个基于servlet容器的轻量灵活的mvc框架,在它整个请求过程中,为了能够灵活定制各种需求,所以提供了一系列的组件完成整个请求的映射,响应等等处理.这里我们来分析下spring ...

  8. web应用启动的时候SpringMVC容器加载过程

    <!-- 配置DispatcherServlet --> <servlet> <servlet-name>springmvc</servlet-name> ...

  9. web.xml中配置启动时加载的servlet,load-on-starup

    web.xml中配置启动时加载的servlet,load-on-starup 使用servlet来初始化配置文件数据: 在servlet的配置当中,<load-on-startup>1&l ...

随机推荐

  1. 获取url后面的参数的方法

    1. function GetRequest() { var url = 'http://wwww.jb51.net/?q=js'; //获取url中"?"符后的字串 if (ur ...

  2. centos下send-mail出错

    当在脚本中执行发邮件操作得时候出现以下错误:send-mail: fatal: parameter inet_interfaces: no local interface found for ::1 ...

  3. css flex 兼容ios android--商品展示 添加购物车

    https://blog.csdn.net/u010035608/article/details/52711248 <!DOCTYPE html> <html> <hea ...

  4. 虚拟机 安装 CentOS7

    初次接触CentOS,最好是在虚拟机中安装: 因为CentOS的安装选项有很多,不理解的情况下千万不要在物理机上尝试: 不然可能出现这种情况:安装好以后,只有黑色的屏幕,只能在里面敲命令:这对于新手来 ...

  5. Windows 2012设置允许单个用户连接多个会话的方法

    WINDOWS 2012 服务器默认只允许单个用户连接一个远程桌面会话,如果已有连接登陆,另外的连接再登陆会踢掉之前的连接.如果需要两个远程桌面同时连接 找到:HKEY_LOCAL_MACHINE\S ...

  6. usb-blaster安装

    插入usb-blaster后,无法安装驱动,一直显示感叹号,更新驱动后显示“文件的哈希值不在指定的目录”这样的错误提示,解决方法如下:1.Windows键+ R,输入shutdown.exe /r / ...

  7. python3使用paramiko操作远程机器

    目标:有A和B两台机器,希望在机器A上操作B上的脚本   解决方法:使用paramiko实现操作远程机器   1.安装paramiko   安装第三方包[pip3 install paramiko] ...

  8. 【Nodejs】ExpressのRequestとResponseの内容

    Request 对象 - request 对象表示 HTTP 请求,包含了请求查询字符串,参数,内容,HTTP 头部等属性.常见属性有: req.app:当callback为外部文件时,用req.ap ...

  9. RxJS之组合操作符 ( Angular环境 )

    一 merge操作符 把多个 Observables 的值混合到一个 Observable 中 import { Component, OnInit } from '@angular/core'; i ...

  10. Netty实践二(心跳检测)

    我们使用Socket通信一般经常会处理多个服务器之间的心跳检测,一般来讲,我们去维护服务器集群,肯定要有一台或几台服务器主机(Master),然后还应该有N台(Slave),那么我们的主机肯定要时时刻 ...