`
drilling_liny
  • 浏览: 2894 次
  • 性别: Icon_minigender_1
  • 来自: 广州
最近访客 更多访客>>
社区版块
存档分类
最新评论

servlet_1

阅读更多

Servlet的简介

Java Servlet是和平台无关的服务器端组件,它运行在Servlet容器(也就是服务器)中。Servlet容器负责Servlet和客户的通信以及调用Servlet的方法,Servlet和客户的通信采用“请求/响应“(http?)的模式。

 

• Servlet可完成如下功能:

– 创建并返回基于客户请求的动态HTML页面。简单地可以理解为在MVC模式中Servlet作为Controler,根据从  model返回的结果,选择将要返回给客户端的JSP(HTML)页面,而这些页面(View),往往会根据从servlet传递的model结果,给客户端呈现出不同的结果,所以就为动态的HTML页面。

– 创建可嵌入到现有 HTML 页面中的部分 HTML 页面(HTML 片段)。

– 与其它服务器资源(如数据库或基于Java的应用程序)进行通信。(主要)

 

Servlet的源码分析

1.每一个servlet都必须要实现Servlet的接口。GenericServlet是一个通用的、不特定于任何协议的Servlet,当然他实现了Servlet接口。而HttpServlet继承了GenericServlet,所以也就实现了Servlet接口。在一般java web开发中,定义Servlet时只需要继承HttpServlet父类即可。

 

2.Servlet接口中定义了一个service方法

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

用于处理对客户端发送的请求,但没有具体实现。HttpServlet对该方法进行了实现,实现方式就是将ServletRequest与ServletRespond转换为HttpServletRequest与HttpServletRespond。

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");
        }
        service(request, response);
}

  

3.注意上方代码最后一句(蓝字) ,当参数转换完毕后,被实现的Servlet接口的service方法会调用HttpServlet重载的service方法

protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException

 

4.在该service方法里,首先获得请求的方法名

(关于方法名在源代码中可注意到:)

private static final String METHOD_DELETE = "DELETE";
private static final String METHOD_HEAD = "HEAD";
private static final String METHOD_GET = "GET";
private static final String METHOD_OPTIONS = "OPTIONS";
private static final String METHOD_POST = "POST";
private static final String METHOD_PUT = "PUT";
private static final String METHOD_TRACE = "TRACE";

 然后,根据方法名去调用对应的doXXX方法,比如说请求为GET,那就去调用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;
            try {
                ifModifiedSince = req.getDateHeader(HEADER_IFMODSINCE);
            } catch (IllegalArgumentException iae) {
                // Invalid date header - proceed as if none was set
                ifModifiedSince = -1;
            }
            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);
    }
}

 

 

   

 5.流程转到doXXX方法以后,如果我们不进行重写doXXX方法,他就会调用父类的doXXX方法,而父类的doXXX方法都是一些会直接抛异常的实现:

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);
        }
}

 其他doXXX方法都是大同小异。所以我们需要重写这些会被调用的方法。

 

 PS:本人使用过的是tomcat服务器,因此以上源代码均来自于tomcat实现的。

 

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics