一个Servlet总是继承自HttpServlet,然后覆写doGet()或doPost()方法。注意到doGet()方法传入了HttpServletRequest和HttpServletResponse两个对象,分别代表HTTP请求和响应。我们使用Servlet API时,并不直接与底层TCP交互,也不需要解析HTTP协议,因为HttpServletRequest和HttpServletResponse就已经封装好了请求和响应。以发送响应为例,我们只需要设置正确的响应类型,然后获取PrintWriter,写入响应即可。
而这样的一个项目最终会打包成一个*.war文件,运行这个文件,需要使用支持ServletAPI的Web容器(Web服务器)。
所以,我们需要一个容器中运行Servlet,比如Tomcat
类似Tomcat这样的服务器也是Java编写的,启动Tomcat服务器实际上是启动Java虚拟机,执行Tomcat的main()方法,然后由Tomcat负责加载我们的*.war文件,并创建一个HelloServlet实例,最后以多线程的模式来处理HTTP请求。如果Tomcat服务器收到的请求路径是/项目名称/hello.do,就转发到HelloServlet并传入HttpServletRequest和HttpServletResponse两个对象。
在通过一个URL路径发起对一个Servlet请求的过程中,其本质是在调用执行Servlet实例的doXXX()方法。该Servlet实例创建和使用的过程,被称为Servlet的生命周期。
整个生命周期包括:实例化、初始化、服务、销毁。
Servlet请求的路径(例如:home.do),查找该Servlet的实例。如果实例不存在,则通过调用构造方法,完成Servlet实例的创建。- // 1.实例化
- public HomeServlet() {
- System.out.println("1.HomeServlet实例被创建");
- }

Servlet的实例,调用init()方法,执行初始化的逻辑。- // 2.初始化
- // 重写HomeServlet父类的init()方法
- @Override
- public void init() throws ServletException {
- System.out.println("2.HomeServlet实例初始化");
- }

Servlet的实例,调用service()方法,如果子类没有重写该方法,则调用HttpServlet父类的service()方法,在父类的该方法中进行请求方式的判断,如果是GET请求,则调用doGet()方法;如果是POST请求,则调用doPost()方法;- // 3.服务
- // 通过实例调用HttpServlet父类的service()方法
- // service()方法中,会根据请求方式(get或post)的不同
- // 调用子类重写的doGet()或doPost()
- // 如果子类没有重写,则相应405
-
- // 根据请求方式的不同,需要调用不同的请求方式
- @Override
- protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
- System.out.println("HomeServlet被GET请求到了"+this.hashCode());
- String method = req.getMethod();
- System.out.println("请求方式:"+method);
-
- System.out.println(req.getRemoteHost());
- }
doGet()方法,则调用子类重写后的doGet()方法; 如果子类没有重写doGet()方法,则调用父类的doGet()方法,在父类的方法实现中,返回一个405状态码的错误页面。doPost()同理
405状态码:代表请求的方式服务器不提供支持。

destroy()方法。- // 4.销毁
-
- @Override
- public void destroy() {
- System.out.println("4.HomeServlet实例被销毁");
- }
关闭服务器:

重启服务器:

Tomcat启动,浏览器发起请求,Servlet生命周期开始,当浏览器第二次发起请求时,不再进行实例化和初始化,通过hashcode考验看出两次是一个实例对象。但是,当服务器关闭或重启时,再次请求,就会产生新的生命周期,进行实例化、初始化、服务、销毁。

为方便大家理解,可以看下图:
