servlet中的service, doGet, doPost方法的区别和联系

大家都知道在javax.servlet.Servlet接口中只有init,service,destroy,getServletConfig,getServletInfo这五个方法.
但是我们在继承HttpServlet的时候为何一般重写doGet和doPost方法呢。

在Servlet中service方法是一直存在的

在Servlet中,service方法是一直存在的,因为最高层的接口Servlet(像HttpServlet等具体的servlet都是直接或者间接实现了这个接口)里面就有这个方法,所以不管是怎样的servlet类,都有service方法,没有service就不能称为一个Servlet了。

service方法一般是不需要重写的

一个servlet被访问时,它默认处理请求的是service方法,这个方法是在servlet接口中定义的。对于service方法,一般来说这个方法是不需要重写的,因为在HttpServlet中已经有了很好的实现,它会根据请求的方式,调用doGet,doPost以及其他的doXXXt方法,也就是说service是用来转向的,所以我们一般写一个servlet,只需要重写doGet或者doPost就可以了。如果重写了service方法,那么servlet容器就会把请求交给这个方法来处理,倘若你重写的service方法没有调用doXXX,即使你在Servlet中又重写了其他doGet doPost等也是不回被调用的,因为Servlet的service被自动调用(就像init destory一样),所以如果你由于某种需要,需要重写service方法,并且更具不同的method调用doPost doGet等时,就要在末尾加上一句super.service(),这样就可以解决问题了。

tomcat实现的service方法的调度请求作用

我们找到tomcat源代码, 可以到官网下载, 找到javax.servlet.http.HttpServlet类, 下面是service方法部分源代码:

protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        String method = req.getMethod();
        if (method.equals("GET")) {
           ...
                this.doGet(req, resp);
           ...
        } else if (method.equals("HEAD")) {
            lastModified = this.getLastModified(req);
            this.maybeSetLastModified(resp, lastModified);
            this.doHead(req, resp);
        } else if (method.equals("POST")) {
            this.doPost(req, resp);
        } else if (method.equals("PUT")) {
            this.doPut(req, resp);
        } else if (method.equals("DELETE")) {
            this.doDelete(req, resp);
        } else if (method.equals("OPTIONS")) {
            this.doOptions(req, resp);
        } else if (method.equals("TRACE")) {
            this.doTrace(req, resp);
        } else {
           ...
        }

    }

注意: JDK只是定义了servlet接口,而实现servlet接口的是比如tomcat等服务器。
从上面的源代码中可以看到,tomcat实现的service方法只是起到了调度请求的作用,
所以我们平时只重写doGet或doPost方法后,会自动按请求类型匹配执行。
如果我们重写了service方法,父类HttpServlet中的service方法就会失效,
所以收到的任何请求都会由我们自己覆写的service方法来处理,
如果我们的servlet中只有service方法,是没有问题的。
但值得注意的是, 如果你同时重写了service和doGet,doPost方法,
一定要在执行完自己代码后调用父类service方法 super.service(),
否自你的doGet和doPost是不会被执行的。

HttpServlet类中的两个service方法

通过API可以看到在HttpServlet类中的两个service方法:service(ServletRequest, ServletResponse)方法是重写的其父类GenericServlet类的方法,这个方法是公共的(public),其作用是接受客户端的请求并将其传递给service(HttpServletRequest, HttpServletResponse)方法;service(HttpServletRequest, HttpServletResponse)方法是HttpServlet类定义的方法,是受保护的(protected),主要作用是接受标准的Http请求(HttpServletRequest),并根据请求方式不同分发到不同的doXXX(HttpServletRequest, HttpServletResponse)方法。 这就是HttpServlet有两个service方法的原因了。一般情况下我们的Servlet只需重写受保护的service方法就够了。

你可能感兴趣的:(java)