glmapper

聊一聊 Web 核心 - Servlet

字数统计: 1.3k阅读时长: 5 min
2018/11/10 Share

Servlet实际上是ServerApplet–小服务程序或服务连接器,用Java编写的服务器端程序,主要功能在于交互式地浏览和修改数据,生成动态Web内容。与常用的协议,如DNS,TCP/IP,HTTP类似,Servlet是作为一整套规范存在的;同时作为J2EE标准的一部分,定义了javaweb开发的标准。Servlet制定了java处理WEB请求的一系列标准,我们只需要按照标准规定的去做就可以了。
实际上,无论是Struts2的FilterDispatcher还是SpringMvc的DispatcherServlet,其底层都是通过实现Sevlet或者Servlet类型的扩展【如:GenericServlet】来实现的。

1.Servlet接口

下图为Servlet3.1中的结构图:

因为Servlet是以规范的方式存在的,实际上就是定义一系列规范接口。在Servlet接口中,主要包括以下几个接口:

  • 1)init方法是在容器启动时被容器调用,且只会被调用一次;
  • 2)getServletConfig方法用于获取ServletConfig;
  • 3)service方法用于处理一个具体的请求
  • 4)getServletInfo方法用于获取Servlet相关的信息:版权等。
  • 5)destroy方法用来销毁一个Servlet,和init一样,只会被调用一次,一般在服务器关闭时用于释放一些资源。

init方法调用时会接受一个ServletConfig类型的参数,用于初始化Servlet,由容器传入。ServletConfig,顾名思义,其包含了Serlvet的配置信息。通常情况下,我们在web.xml文件中定义Serlvet时,会通过init-param标签来进行参数配置。在Springmvc的配置中,通常通过以下方式来配置参数:

2.ServletConfig接口


1)getServletName用于获取Servlet的名字,也就是我们在web.xml中定义的servlet-name
2)getServletContext返回ServletContext,代表我们当前应用本身
3)getInitParameter用于获取init-param配置的参数
4)getInitParameterNames用于获取所有init-param配置名字的集合
ServletContext和ServletConfig最常见的使用就是传递初始化参数。来看下spring中的contextConfigServlet的参数配置

通过context-param配置的contextConfigLocation配置到了ServletContext中,再通过Servlet下的init-param配置的contextConfigLocation配置到ServletConfig中,在Servlet中可以通过getInitParameter方法获取具体的信息。

3.GenericServlet

GenericServlet是Servlet的默认实现,代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76

package javax.servlet;
import java.io.IOException;
import java.io.Serializable;
import java.util.Enumeration;
import java.util.ResourceBundle;
public abstract class GenericServlet
implements Servlet, ServletConfig, Serializable
{
private static final String LSTRING_FILE = "javax.servlet.LocalStrings";
private static ResourceBundle lStrings = ResourceBundle.getBundle("javax.servlet.LocalStrings");
private transient ServletConfig config;
public void destroy()
{
}
public String getInitParameter(String name)
{
ServletConfig sc = getServletConfig();
if (sc == null) {
throw new IllegalStateException(lStrings.getString("err.servlet_config_not_initialized"));
}
return sc.getInitParameter(name);
}
public Enumeration getInitParameterNames()
{
ServletConfig sc = getServletConfig();
if (sc == null) {
throw new IllegalStateException(lStrings.getString("err.servlet_config_not_initialized"));
}
return sc.getInitParameterNames();
}
public ServletConfig getServletConfig()
{
return this.config;
}
public ServletContext getServletContext()
{
ServletConfig sc = getServletConfig();
if (sc == null) {
throw new IllegalStateException(lStrings.getString("err.servlet_config_not_initialized"));
}
return sc.getServletContext();
}
public String getServletInfo()
{
return "";
}
public void init(ServletConfig config)
throws ServletException
{
this.config = config;
init();
}
public void init()
throws ServletException
{
}
public void log(String msg)
{
getServletContext().log(getServletName() + ": " + msg);
}
public void log(String message, Throwable t)
{
getServletContext().log(getServletName() + ": " + message, t);
}
public abstract void service(ServletRequest paramServletRequest, ServletResponse paramServletResponse)
throws ServletException, IOException;
public String getServletName()
{
ServletConfig sc = getServletConfig();
if (sc == null) {
throw new IllegalStateException(lStrings.getString("err.servlet_config_not_initialized"));
}
return sc.getServletName();
}
}

从其继承和实现关系来看,GenericServlet主要做了3件事:
1.实现了ServletConfig接口,这样我们就可以直接调用ServletConfig里面的方法;
GenericServlet实现了ServletConfig,可以在需要的时候直接调用ServletConfig中的方法,不需要再先获取ServletConfig对象;比如,获取ServletContext的时候可以直接调用getServletContext,而无需调用getServletConfig().getServletContext(),但是实际上,其底层的内部实现还是在内部还是进行了getServletConfig().getServletContext()的调用。

2.提供了无参的init方法
GenericServlet实现了Servlet的init(ServletConfig config)方法,在里面将config设置给了其内部变量config,然后调用了无参的init方法;此方法可以在子类中通过覆盖它来完成初始化工作。

这种方式具有的有点包括以下几点:
a.config设置为内部属性,这样可以在ServletConfig的接口方法中直接调用Config的相应方法来执行;
b.我们在写Serlvet的时候可以不用再关心Config,只需要执行自己的初始化逻辑即可
c.在重写init方法时,不需要再调用super.init(config)。
3.提供了Log方法
GenericServlet提供了2个log方法,一个用于记录日志,一个用于记录异常。其具体的实现是通过传给ServletConfig的日志实现的。
GenericServlet是与具体协议无关的。

4.HttpServlet

HttpServlet是基于Http协议实现的Servlet的基类,写Servlet时直接继承HttpServlet即可,不需要再重头实现Servlet接口,SpringMvc中的dispatcherServlet就是HttpServlet的子类。 HttpServlet是与Http协议相关的,HttpServlet处理请求主要是通过重写父类的service方法来完成具体的请求处理的。在service方法中首先是将ServletRequest和ServletResponse转换成HttpServletRequest和HttpServletResponse,然后根据请求的不同路由到不同的处理过程中去【处理方法就是我们常见的doXXX的方法。最常见的就是doGet和doPost】

原文作者:GuoLei Song

原文链接:http://www.glmapper.com/2018/11/10/topic-servlet/

发表日期:November 10th 2018, 1:38:37 pm

更新日期:December 14th 2019, 11:08:59 am

版权声明:转载请注明出处

CATALOG
  1. 1. 1.Servlet接口
  2. 2. 2.ServletConfig接口
  3. 3. 3.GenericServlet
  4. 4. 4.HttpServlet