Servlet源码深入学习

Servlet框架由两个java包组成:

javax.servlet包:定义了所有的Servlet类都必须实现或扩展的通用接口和类。

javax.servlet.http包:定义了采用HTTP协议通信的HttpServlet类。

Servlet生命周期方法:init()、service()、destroy()。

init():Servlet容器启动时自动装载某些Servlet(在web.xml中的Servlet元素的子元素<load-on-startup>中配装载顺序);Servlet容器启动后,客户首次向Servlet发出请求;Servlet类文件更新后,重新装载Servlet。Servlet被装载后,Servlet容器创建一个servlet实例且调用init()方法,在整个生命周期中,init()只调用一次。
service():响应请求。
destroy():web应用被终止;servlet容器终止;servlet容器重新装载servlet新实例。

对于到达Servlet容器的请求,Servlet容器创建特定于该请求的ServletRequest对象和ServletResponse对象,然后调用Servlet的service方法。service方法从ServletRequest对象获得请求信息并处理该请求,然后通过ServletResponse对象响应结果。

每一个Servlet都必须实现Servlet接口,GenericServlet是个通用的、不特定于任何协议的Servlet接口,它实现了Servlet接口,而HttpServlet继承了GenericServlet,因此HttpServlet也实现了Servlet接口,所以我们定义的Servlet只需继承HttpServlet即可。

Servlet接口定义了一个service方法,HttpServlet对该方法进行了实现,实现方式是将ServletRequest与ServletResponse转换为HttpServletRequest与HttpServletResponse,如下
public void service(ServletRequest req, ServletResponse res)
        throws ServletException, IOException {

        HttpServletRequest  request;
        HttpServletResponse response;
        
        try {
            request = (HttpServletRequest) req;
            response = (HttpServletResponse) res;
        } catch (ClassCastException e) {
            throw new ServletException("non-HTTP request or response");
        }
       //HttpServlet自己的service方法
        service(request, response);
    }

HttpServlet定义了自己的service方法,首先获得请求方法的方法名,然后根据方法名调用对应的doXXX方法,如doGet方法。

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 / 1000 * 1000)) {
                    // 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);
        }
    }

对于每个doXXX方法的实现,HttpServlet都是返回错误信息,所有要重写这些doXXX方法。如HttpServlet对doGet的实现

protected void doGet(HttpServletRequest req, HttpServletResponse resp)
        throws ServletException, IOException
    {
        String protocol = req.getProtocol();
        String msg = lStrings.getString("http.method_get_not_supported");
        if (protocol.endsWith("1.1")) {
            resp.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, msg);
        } else {
            resp.sendError(HttpServletResponse.SC_BAD_REQUEST, msg);
        }
    }

Servlet容器启动时为每个应用创建唯一的一个ServletContext,它相当于JSP页面的内置对象application,可以把这个ServletContext看成是一个Web应用的服务器组件的共享内存。它有存取该内存数据的两个方法:setAttribute(String name, Object obj)和getAttribute(String name)。

对于Servlet的同步问题,通常我们去除成员变量,使用局部变量。不推荐使用同步代码块。

Cookies 由servlet容器response到客户端。

 

你可能感兴趣的:(servlet)