欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Servlet的体系结构

程序员文章站 2022-06-08 21:36:46
...


Servlet的体系结构

编写一个Servlet的步骤:

1、其实我们要写一个Servlet只要写一个类去实现Servlet就可以了,但是,Servlet里面有很多的方法,我们没有必要每次为了写一个Servlet而去实现这么多方法。
2、所以这里提供了一个GenericServlet,这个类实现了一些方法,我们可以直接继承这个类还不用去实现Servlet,但是我们要做的web,实现Http协议。GenericServlet里面并没有有关协议的一些方法。
3、继承自GenericServlet类的HttpServlet这个类里面有http协议的方法,里面有doGet()和doPost()方法,我们应该联想到get和post请求。

Servlet接口:

在Servlet接口中定义了3个与生命周期相关的方法。同时也是最重要的三个方法。

init():当容器初始化Servlet时调用该方法。在一个生命周期中只被执行一次,调用service方法前,初始化必须完成;
service():当客户端请求访问Servlet时,容器调用该方法。它是Servlet的核心,负责响应客户的请求。
destroy():当容器销毁Servlet时调用该方法。仅执行一次,在服务器端停止且卸载Servlet时执行该方法。

GenericServlet类:

通过分析GenericServlet源码:

1、在GenericServlet中只是对Servlet中的有参的init方法实现了空实现,并调用了无参的init方法。

 public void init(ServletConfig config) throws ServletException {
	this.config = config;
	this.init();
    }
   public void init() throws ServletException {
    }

2、在GenericServlet中只是对Servlet中的service方法实现了空实现

public abstract void service(ServletRequest req, ServletResponse res)
	throws ServletException, IOException;

3、在GenericServlet中也紧是对Servlet中的destroy方法进行了空实现。

public void destroy() {

}

HttpServlet类:

通过分析HttpServlet源码

1、HttpServlet所实现的init方法是GenericServlet已经实现的。
2、GenericServlet根据请求方式get或者post,并相应的调用doGet或者doPost方式。

public void service(ServletRequest req, ServletResponse res)
        throws ServletException, IOException
    {
        HttpServletRequest  request;
        HttpServletResponse response;
        
        if (!(req instanceof HttpServletRequest &&
                res instanceof HttpServletResponse)) {
            throw new ServletException("non-HTTP request or response");
        }

        request = (HttpServletRequest) req;
        response = (HttpServletResponse) res;

        service(request, response);
    }
    protected void service(HttpServletRequest req, HttpServletResponse resp)
        throws ServletException, IOException
    {
        String method = req.getMethod();

        if (method.equals(METHOD_GET)) {
            long lastModified = getLastModified(req);
            if (lastModified == -1) {
                // servlet doesn't support if-modified-since, no reason
                // to go through further expensive logic
                doGet(req, resp);
            } else {
                long ifModifiedSince = req.getDateHeader(HEADER_IFMODSINCE);
                if (ifModifiedSince < lastModified) {
                    // If the servlet mod time is later, call doGet()
                    // Round down to the nearest second for a proper compare
                    // A ifModifiedSince of -1 will always be less
                    maybeSetLastModified(resp, lastModified);
                    doGet(req, resp);
                } else {
                    resp.setStatus(HttpServletResponse.SC_NOT_MODIFIED);
                }
            }

        } else if (method.equals(METHOD_HEAD)) {
            long lastModified = getLastModified(req);
            maybeSetLastModified(resp, lastModified);
            doHead(req, resp);

        } else if (method.equals(METHOD_POST)) {
            doPost(req, resp);
            
        } else if (method.equals(METHOD_PUT)) {
            doPut(req, resp);
            
        } else if (method.equals(METHOD_DELETE)) {
            doDelete(req, resp);
            
        } else if (method.equals(METHOD_OPTIONS)) {
            doOptions(req,resp);
            
        } else if (method.equals(METHOD_TRACE)) {
            doTrace(req,resp);
            
        } else {
            //
            // Note that this means NO servlet supports whatever
            // method was requested, anywhere on this server.
            //

            String errMsg = lStrings.getString("http.method_not_implemented");
            Object[] errArgs = new Object[1];
            errArgs[0] = method;
            errMsg = MessageFormat.format(errMsg, errArgs);
            
            resp.sendError(HttpServletResponse.SC_NOT_IMPLEMENTED, errMsg);
        }
    }

3、HttpServlet所实现的destroy方法是GenericServlet已经实现的。

综上所述分析:

  当servlet对象创建后会调用有参数的init方法。在有参数的init方法中又调用了无参数的init方法。当需要编写init方法的时候,我们只需要重写无参数的init方法就可以。
  继承自HttpServlet的时候我们不需要重写service方法,request = (HttpServletRequest) req; response = (HttpServletResponse) res;将serivce方法分流。我们只需要重写doGet方法和doPost方法
在HttpServlet中serivce的源码中我们可以看到利用doGet和doPost进行分流操作。String method = req.getMethod();根据method的值我们可以判断调用doGet还是doPost方式。

扩展:Servlet的自动加载

<servlet>
<servlet-name>Hello</servlet-name>
<servlet-class>com.syj.Servlet.MyServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
  • 可以让Servlet随着服务器的启动而自动加载
  • <load-on-startup>X</load-on-startup>
  • 其中X代表1-10数字越小代表的优先级越高,1优先级最高,优先级高的先加载。