深入学习(一)—— Servlet的运行流程和源码分析

现在市面上的框架越来越多,从SSH到SSM,再到现在的SpringBoot,微服务,技术更新迭代的速度之快,让学习者都有点应接不暇,有点被技术鞭策着,去学习最新技术的感觉。很多人,在追逐框架技术的道路上会或多或少有些迷失,包括我自己,在追逐框架技术的道路上停不下来,刹不住车。该给自己降降温了,该回头把最基本,最重要的基础-JavaWeb重温一遍了。

学习了这么多框架,从SSH、SSM再到SpringBoot,学会了如何使用,学会了编写CRUD,但都不愿意去啃运行流程、运行源码这块硬骨头。然而,学的越多,越感觉有点底气不足,有点违背了万丈高楼平地起的道理。接下来,就总结一系列关于JavaWeb基础、SSH、SSM运行原理和源码分析的博文,来鞭策自己,让自己慢下来,静下心来。

Servlet

Servlet是JavaWeb里最重要的内容了,这一块所包含的知识点,也是非常的多的。Servlet为创建基于web的应用程序提供了基于组件、独立于平台的方法。

servlet实际上就是一个类,是基于Java服务端的java类,在servlet容器的基础上,servlet就可以在服务端进行运行。

以下是servlet-api.jar的目录结构,servlet-api.jar在tomcat安装目录下的lib文件夹中:

为了方便查看源码,这里直接在IDEA中查看源码。

让我们来看看HelloWorld类的Diagram图。

可以看到,HelloWorld这个Servlet的关系图有:

  • 两个顶级接口
    • Servlet
    • ServletConfig
  • 接口的实现类
    • GenericServlet
  • 基于HTTP协议的实现类
    • HttpServlet

 

 顶级接口:Servlet

package javax.servlet;
import java.io.IOException;

public interface Servlet {
    void init(ServletConfig var1) throws ServletException;

    ServletConfig getServletConfig();

    void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException;

    String getServletInfo();

    void destroy();
}

 抽象类:GenericServlet

package javax.servlet;

import java.io.IOException;
import java.io.Serializable;
import java.util.Enumeration;

public abstract class GenericServlet implements Servlet, ServletConfig, Serializable {
    
    ...

    public GenericServlet() {
    }

    public void destroy() {
    }

    ...

    public void init(ServletConfig config) throws ServletException {
        this.config = config;
        this.init();
    }

    public void init() throws ServletException {
    }

    public abstract void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException;

}

 实现类:HttpServlet

package javax.servlet.http;
...

public abstract class HttpServlet extends GenericServlet {

    ...    

    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(405, msg);
        } else {
            resp.sendError(400, msg);
        }

    }

    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        String protocol = req.getProtocol();
        String msg = lStrings.getString("http.method_post_not_supported");
        if (protocol.endsWith("1.1")) {
            resp.sendError(405, msg);
        } else {
            resp.sendError(400, msg);
        }

    }

    protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        String method = req.getMethod();
        long lastModified;
        if (method.equals("GET")) {
            lastModified = this.getLastModified(req);
            if (lastModified == -1L) {
                this.doGet(req, resp);
            } else {
                long ifModifiedSince;
                try {
                    ifModifiedSince = req.getDateHeader("If-Modified-Since");
                } catch (IllegalArgumentException var9) {
                    ifModifiedSince = -1L;
                }

                if (ifModifiedSince < lastModified / 1000L * 1000L) {
                    this.maybeSetLastModified(resp, lastModified);
                    this.doGet(req, resp);
                } else {
                    resp.setStatus(304);
                }
            }
        } 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 {
            String errMsg = lStrings.getString("http.method_not_implemented");
            Object[] errArgs = new Object[]{method};
            errMsg = MessageFormat.format(errMsg, errArgs);
            resp.sendError(501, errMsg);
        }

    }

    /* 服务器先调用这个公共的service方法,然后进行request和response强转,然后再调用重载的service方法 */
    public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {
        HttpServletRequest request;
        HttpServletResponse response;
        try {
            request = (HttpServletRequest)req;
            response = (HttpServletResponse)res;
        } catch (ClassCastException var6) {
            throw new ServletException("non-HTTP request or response");
        }

        this.service(request, response);
    }    

}

HelloWorld.java

package com.test;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;

public class HelloWorld extends HttpServlet {
    private String msg;

    /**
     * 初始化过程
     * @throws ServletException
     */
    public void init() throws ServletException {
        msg = "Hello Servlet!";
    }

    @Override
    public void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //设置响应内容类型
        resp.setContentType("text/html");

        //业务逻辑
        PrintWriter out = resp.getWriter();
        out.print("<h1>" + msg + "</h1>");

    }

    @Override
    public void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        this.doGet(req, resp);
    }
}

 

 

先简单的说一下本地运行的Servlet运行的流程:

① 在浏览器输入url,发送请求 ——> 请求发送给服务器 ——> 服务器读取web.xml中的servlet,匹配请求url

——> ④ 进入servlet生命周期 ——> 调用service()方法 ——> ⑥ 根据请求方式调用doGet或doPost方法

——> 处理业务逻辑,返回json/html/xml

每步流程的粗略总结:

①~②:在浏览器中输入url,敲回车。浏览器发送请求,包括请求方法URI协议、版本,请求头,请求正文。然后请求发送至tomcat服务器,tomcat服务器收到请求后,会创建一个request对象,然后把收到的请求信息封装到request对象里面。

:tomcat加载完web.xml后,会等待用户发送请求。当tomcat服务器收到用户发送的请求后,会调用一个mapper的internalMapWrapper方法,这方法是org.apache.tomcat.util.http.mapper.Mapper类中的一个方法。然后找到对应url-pattern的servlet。

:进入servlet生命周期(servlet的init()方法早在servlet容器也就是tomcat启动的时候就调用了,并且只调用一次。)然后调用service()方法来处理客户端的请求。

⑤~⑥:调用service()方法,在该方法中,将ServletRequest和ServletResponse强转为HttpServletRequest和HttpServletResponse,然后重载service()方法。在重载service()方法里,会根据请求方式来判断是调用doGet()还是doPost()方法。

:在对应的doGet()或doPost()方法里处理业务逻辑,返回json/html/xml作为响应。这个响应会返回给tomcat服务器,然后tomcat服务器再对这个响应进行包装、渲染,并以HTTP响应的形式发送给客户端。

如果tomcat服务器关闭或重启,servlet就被destroy,然后被垃圾回收器进行垃圾回收。

 

总结一下Servlet的三种实现方式:

  1. 实现顶级Servlet接口
  2. 继承抽象类GenericServlet
  3. 继承抽象类HttpServlet

需要注意的是,顶级Servlet和抽象类GenericServlet都只有service方法,所以当收到客户端发送过来的请求后,需要判断是POST还是GET请求。

 

Servlet容器

什么是Servlet容器呢?Servlet容器实际上就是tomcat。tomcat的一个重要作用就是作为Servlet容器,用于在发送的请求和响应之上提供网络服务,解码基于MIME的请求,格式化MIME的响应。Servlet必须部署到Servlet容器中,由容器来实例化Servlet和收到请求后调用Servlet的方法(例如doPost和doGet方法)。

以Servlet容器的角度来看看客户端发送请求,服务端响应的过程。

用户通过单击某个链接或者直接在浏览器的地址栏中输入URL来访问Servlet,Web服务器接收到该请求后,并不是将 请求直接交给Servlet,而是交给Servlet容器。Servlet容器实例化Servlet,调用Servlet的一个特定方法对请求进行处理, 并产生一个响应。这个响应由Servlet容器返回给Web服务器,Web服务器包装这个响应,以HTTP响应的形式发送给Web浏览器。

总结一下Servlet容器的作用:

提供作用作用内容
通信支持利用容器提供的方法,你能轻松的让servlet与web服务器对话,而不用自己建立serversocket、监听某个端口、创建流等 等。容器知道自己与web服务器之间的协议,所以你的servlet不用担心web服务器(如Apache)和你自己的web代码之间的API,只需要考 虑如何在servlet中实现业务逻辑(如处理一个订单)
生命周期管理servlet容器控制着servlet的生与死,它负责加载类、实例化和初始化servlet,调用servlet方法,以及使servlet实例被垃圾回收,有了servlet容器,你不需要太多的考虑资源管理。
多线程支持容器会自动为它所接收的每个servlet请求创建一个新的java线程。针对用户的请求,如果servlet已经运行完相应的http服务方法,这个线程就会结束。这并不是说你不需要考虑线程安全性,其实你还会遇到同步问题,不过这样能使你少做很多工作。
声明方式实现安全利用servlet容器,你可以使用xml部署描述文件来配置和修改安全性,而不必将其硬编码写到servlet类代码中。
JSP支持servlet容器负责将jsp代码翻译为真正的java代码。

 

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Servlet 是 JavaEE 规范中的一部分,是处理 Web 请求的组件。Servlet 运行在服务器端,能够接收客户端发来的请求,并给客户端响应结果。下面我们来看看 Servlet 的基本概念和使用方法。 ## 一、Servlet 的基本概念 ### 1.1 Servlet 的生命周期 Servlet 的生命周期包含以下三个阶段: - 初始化阶段(init):当 Servlet 实例化后,Web 容器会调用其 init() 方法进行初始化操作。在此阶段Servlet 可以执行一些初始化操作,例如读取配置信息、建立数据库连接等。 - 请求处理阶段(service):当客户端发来请求时,Web 容器会创建一个线程调用 Servlet 的 service() 方法处理请求。在此阶段Servlet 可以获取请求参数、处理请求并生成响应数据。 - 销毁阶段(destroy):当 Web 应用停止或 Servlet 被卸载时,Web 容器会调用 Servlet 的 destroy() 方法进行清理工作。在此阶段Servlet 可以释放资源、关闭数据库连接等。 ### 1.2 Servlet 的配置 在使用 Servlet 时,需要在 web.xml 文件中进行配置。以下是一个 Servlet 的基本配置: ```xml <servlet> <servlet-name>MyServlet</servlet-name> <servlet-class>com.example.MyServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>MyServlet</servlet-name> <url-pattern>/myservlet</url-pattern> </servlet-mapping> ``` 其中,servlet-name 表示 Servlet 的名称,servlet-class 表示 Servlet 的类名,url-pattern 表示请求的 URL 匹配规则。 ## 二、Servlet 的使用方法 ### 2.1 编写 Servlet 编写 Servlet 有两种方法:一种是继承 HttpServlet 类,另一种是实现 Servlet 接口。这里以继承 HttpServlet 类为例: ```java public class MyServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 处理 GET 请求 PrintWriter out = resp.getWriter(); out.println("Hello, world!"); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 处理 POST 请求 PrintWriter out = resp.getWriter(); out.println("Hello, world!"); } } ``` 在 Servlet 中,doGet() 方法用于处理 GET 请求,doPost() 方法用于处理 POST 请求。通过调用 HttpServletResponse 对象的 getWriter() 方法可以向客户端返回响应数据。 ### 2.2 部署 Servlet 将编写好的 Servlet 部署到 Web 容器中,有两种方法:一种是将 Servlet 类打成 war 包放到 Web 容器的 webapps 目录下,另一种是通过 Eclipse 等开发工具将 Servlet 部署到 Web 容器中。部署完成后,可以通过访问 Servlet 的 URL 来测试 Servlet 是否正常工作。 ## 三、总结 本文介绍了 Servlet 的基本概念和使用方法。Servlet 是 Java Web 开发中非常重要的组件,掌握 Servlet 的使用方法对于 Java Web 开发人员来说是必不可少的。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值