Servlet容器在收到客户机请求的时候,首先会解析请求url,根据web.xml的配置,去找到对应的servlet。如果没有找到对应的servlet,服务器有一个默认Servlet的配置项,位于$CATALINA_HOME/conf/web.xml中被全局声明(也就是说你项目下的web.xml文件里没有)。容器会检查请求的servlet是否被实例化,如果没有,创建一个servlet实例:
容器会调用过滤器ApplicationFilterChain的doFilter(ServletRequest request, ServletResponse response)方法,源码在此省略,里面有两处截图如下:
--------------------------------------------------------------------
然后,分两种可能。第一种是没有对应的servlet,此时,容器会先调用HttpServlet的service方法,在这里将ServletRequest对象和ServletResponse对象强转为HttpServletRequest对象和HttpServletResponse对象,再调用DefaultServlet实例的service方法,具体代码稍微复杂,但是如果是不存在的servlet,则会响应404。第二种是有对应的servlet,容器则先调用HttpServlet的service方法,在这里将ServletRequest对象和ServletResponse对象强转为HttpServletRequest对象和HttpServletResponse对象,再调用实例的service方法,倘若程序员重写了service方法,则按照程序员写的来。倘若程序员没有重写,则调用父类(HttpServlet类)的service方法:
而默认的service方法,会根据请求的类型,调用不同的方法。
其实,这些do开头的方法,也都是protected方法,换句话说,也是需要程序员重写的。如果客户端发来一次post请求,程序员既没有重写service(HttpServletRequest req, HttpServletResponse resp)方法,也没有重写doPost(HttpServletRequest req, HttpServletResponse resp)方法,那么就调用默认的doPost方法,如下所示:
可以看到,在默认的doPost方法中,Tomcat容器根据协议的类型,给客户机提示405或者400。如果程序员重写了doPost方法,那么就按照程序员写的执行,如果没有重写,则响应405或者400。