Servlet相关接口和Servlet的生命周期

时间:2022-04-10 10:13:36

创建一个Servlet类最直接的方式是实现javax.servlet.Servlet接口,Servlet接口的定义如下,其中定义了5个方法,Servlet的生命周期就是通过这几个方法体现的:

 1 public interface Servlet {
 2     public void init(ServletConfig config) throws ServletException;
 3     
 4     public ServletConfig getServletConfig();
 5     
 6     public void service(ServletRequest req, ServletResponse res)
 7             throws ServletException, IOException;
 8             
 9     public String getServletInfo();
10     
11     public void destroy();
12 }

1 . 加载和实例化

首先定义一个DemoServlet实现Servlet接口,覆写接口中的方法,分别打印各自的方法名,在无参构造方法中也打印一句话:

 1 public class DemoServlet implements Servlet {
 2 
 3     public DemoServlet() {
 4         System.out.println("constructor…");
 5     }
 6 
 7     @Override
 8     public void destroy() {
 9         System.out.println("DemoServlet.destroy()");
10     }
11 
12     @Override
13     public ServletConfig getServletConfig() {
14         System.out.println("DemoServlet.getServletConfig()");
15         return null;
16     }
17 
18     @Override
19     public String getServletInfo() {
20         System.out.println("DemoServlet.getServletInfo()");
21         return null;
22     }
23 
24     @Override
25     public void init(ServletConfig config) throws ServletException {
26         System.out.println("DemoServlet.init()");
27     }
28 
29     @Override
30     public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {
31         System.out.println("DemoServlet.service()");
32     }
33 
34 }

Servlet要被容器所管理,需要在web.xml中进行配置:

1 <servlet>                                           
2     <servlet-name>demo</servlet-name>               
3     <servlet-class>servlet.DemoServlet</servlet-classs>
4 </servlet>                                          
5 <servlet-mapping>                                   
6     <servlet-name>demo</servlet-name>               
7     <url-pattern>/demo</url-pattern>                
8 </servlet-mapping>                                  

启动服务器,通过“http://localhost:8080/${contextpath}/demo”请求该Servlet,控制台打印输出如下:

Servlet相关接口和Servlet的生命周期

刷新页面,再次请求该Servlet,控制台打印如下,可以看到,第二次请求只执行了service(ServletRequest req, ServletResponse res)方法:

Servlet相关接口和Servlet的生命周期

当服务器接收到请求时,会通过web.xml中的配置寻找匹配的Servlet,找不到则报404错误;找到之后,根据配置的Servlet类的全限定名查找服务器中是否已经存在该Servlet类的实例,若存在则直接使用它来处理本次请求(上面的第二次请求,该Servlet实例已经存在,直接调用该实例的service(ServletRequest req, ServletResponse res)方法。);若不存在,则根据全限定名通过反射创建实例(调用公共无参的构造方法)。从这里可以看出,若没有提供可访问的无参构造方法,会报错:

1 //提供有参构造方法,覆盖了默认的无参构造方法                     
2 public DemoServlet(String s) {              
3     System.out.println("constructor…");     
4 }                                           

 Servlet相关接口和Servlet的生命周期

服务器通过反射调用Class.newInstance()方法创建实例,找不到可访问的无参构造方法,自然会报错。

2 . 初始化

从上面可以看到,在第一次请求,创建Servlet实例的时候,调用了init(ServletConfig config)方法完成初始化。初始化只会在创建实例的时候进行,以后的请求不会再进行初始化,直接调用service()方法处理请求。服务器在进行Servlet初始化的时候,会首先准备一个ServletConfig对象,通过该对象可以从应用程序的配置信息中获取初始化的参数信息;数据库连接的建立等任务通常也是在初始化阶段完成。

ServletConfig接口中定义了4个方法,用来获取初始化参数信息:

 1 public interface ServletConfig {
 2     //获取Servlet的名字,即在web.xml中<servlet-name>demo</servlet-name>节点配置的信息
 3     public String getServletName();
 4 
 5     //获取Servlet上下文对象
 6     public ServletContext getServletContext();
 7 
 8     //根据参数名获取初始化参数值
 9     public String getInitParameter(String name);
10     
11     //获取所有初始化参数名
12     public Enumeration<String> getInitParameterNames();
13 }

初始化参数可以在web.xml中配置:

 1 <servlet>                                            
 2     <servlet-name>demo</servlet-name>                
 3     <servlet-class>servlet.DemoServlet</servlet-class>
 4     <init-param>                                     
 5         <param-name>username</param-name>            
 6         <param-value>admin</param-value>             
 7     </init-param>                                    
 8 </servlet>                                           
 9 <servlet-mapping>                                    
10     <servlet-name>demo</servlet-name>                
11     <url-pattern>/demo</url-pattern>                 
12 </servlet-mapping>                                   

注意:如果在web.xml中配置了<load-on-startup>1</load-on-startup>,该Servlet会在服务器启动的时候就完成实例的创建和初始化,而不是在第一次请求的时候:

1 <servlet>                                                 
2     <servlet-name>demo</servlet-name>                     
3     <servlet-class>servlet.DemoServlet</servlet-class>    
4     <load-on-startup>1</load-on-startup>                  
5 </servlet>                                                
6 <servlet-mapping>                                         
7     <servlet-name>demo</servlet-name>                     
8     <url-pattern>/demo</url-pattern>                      
9 </servlet-mapping>                                        

Servlet相关接口和Servlet的生命周期

3 . 处理请求

服务器调用service(ServletRequest req, ServletResponse res)方法来处理请求。处理请求之前,init()方法必须成功执行。服务器在调用service(ServletRequest req, ServletResponse res)方法之前,已经准备好了ServletRequest对象和ServletResponse对象,并作为参数传给service(ServletRequest req, ServletResponse res)方法。

ServletRequest接口和ServletResponse接口中定义了用来处理请求和响应的相关方法:

Servlet相关接口和Servlet的生命周期Servlet相关接口和Servlet的生命周期

4 . 销毁

当容器检测到一个Servlet实例应该从服务中被移除的时候,容器就会调用实例的destroy()方法,以便让该实例可以释放它所使用的资源,保存数据到持久存储设备中。当需要释放内存或者容器关闭时,容器就会调用Servlet实例的destroy()方法。在destroy()方法调用之后,容器会释放这个Servlet实例,该实例随后会被Java的垃圾收集器所回收。如果再次需要这个Servlet处理请求,Servlet容器会创建一个新的Servlet实例。

destroy()方法跟init()方法一样,在Servlet的生命周期中只会被执行一次,但destroy()方法不会总是被执行,如服务器被异常终止时,destroy()方法就不能被执行。