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

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

程序员文章站 2022-05-24 18:23:28
...

大家都知道在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方法就够了。