JSP 编译原理:JSP 是 Servlet?如何用 Eclipse 查看 JSP 编译生成的 Servlet 源文件?


前言

相信大家都了解,JSP 页面在请求的时候会先被 Tomcat 编译为 Servlet(Servlet 是用 Java 语言编写的服务器端程序),然后再由 Java 编译器编译为以 .class 结尾的中间字节码文件,最后再编译为机器能识别的二进制机器码文件。我们通过使用 Eclipse 演示一个小案例,了解 JSP 编译原理的同时来帮大家找到并剖析编译后生成的 Servlet 的 Java 代码文件。

在这里插入图片描述


声明:本文由作者“白鹿第一帅”于 CSDN 社区原创首发,未经作者本人授权,禁止转载!爬虫、复制至第三方平台属于严重违法行为,侵权必究。亲爱的读者,如果你在第三方平台看到本声明,说明本文内容已被窃取,内容可能残缺不全,强烈建议您移步“白鹿第一帅” CSDN 博客查看原文,并在 CSDN 平台私信联系作者对该第三方违规平台举报反馈,感谢您对于原创和知识产权保护做出的贡献!

文章作者白鹿第一帅作者主页https://blog.csdn.net/qq_22695001,未经授权,严禁转载,侵权必究!

一、JSP 文件编译流程原理

JSP 页面在请求的时候会先被 Tomcat 编译为 Servlet(Servlet 是用 Java 语言编写的服务器端程序),然后再由 Java 编译器编译为以 .class 结尾的中间字节码文件,最后再编译为机器能识别的二进制机器码文件,整体流程如下图所示:

在这里插入图片描述

二、创建并运行待测试 JSP 页面

我们先创建一个动态 Web 项目 JavaWebDemo_2020,并创建好一个 JSP 页面 Demo01.jsp,在 Tomcat 服务器下运行一次,我们的测试代码如下:

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
	<h1>Hello,bailu!</h1>
</body>
</html>

运行结果,具体如下图所示:

在这里插入图片描述

三、查找 JSP 编译文件输出位置

现在我们的项目已经在服务器运行了一次,按照上面一中所述,JSP 页面已经完成了编译流程并已经输出,那么我们怎么样才能找到输出文件呢?

3.1、打开动态项目运行配置

在当前项目下点击 Run As→Run Configurations…进入运行配置页面,具体如下图所示:

在这里插入图片描述
点击你当前使用的的服务器,我的是:Tomcat v9.0 Server at localhost,我们接着点击 Arguments,具体如下图所示:

在这里插入图片描述

3.2、查看 JSP 编译文件输出位置

根据 Tomcat 虚拟机参数信息查看编译文件输出位置,第一条数据 Dcatalina.base 即为 JSP 文件编译后的输出目录,比如我的输出目录即为:D:\bailu\eclipse-jee-2019-09-R-win32-x86_64\eclipse\eclipse-workspace.metadata.plugins\org.eclipse.wst.server.core\tmp0,具体如下图所示:

在这里插入图片描述

3.3、查看 JSP 编译输出文件

我们复制编译文件输出目录,在“我的电脑”打开该路径,出现如下目录结构,具体如下图所示:

在这里插入图片描述
我们根据 JSP 关于 Web 服务目录的基础知识,可以明确,编译输出文件在 work 文件夹中,打开该文件夹最底层文件夹,我们可以看到刚才在 Eclipse 中运行的当前项目 JavaWebDemo_2020 的输出文件夹,具体如下图所示:

在这里插入图片描述
我们顺着项目文件夹逐级往下查看,就可以看到我们刚才运行 Demo01.jsp 页面的编译输出文件,一个是 JSP 初次编译生成的 .java 文件,一个是 java 文件编译后生成的 .class 中间字节码文件,具体如下图所示:

在这里插入图片描述

四、JSP 编译输出 Servlet 的论证

见证奇迹的时候到了!
在这里插入图片描述
我们使用 IDE 打开 JSP 页面编译生成的 .java 文件(.java 文件的可读性与.class 文件强得多),一行一行与上面我们的 JSP 页面对比,是不是一样?这就直接可以说明,该 java 文件就是 JSP 页面编译后生成的,具体代码如下:

/*
 * Generated by the Jasper component of Apache Tomcat
 * Version: Apache Tomcat/9.0.35
 * Generated at: 2020-10-28 07:21:14 UTC
 * Note: The last modified time of this file was set to
 *       the last modified time of the source file after
 *       generation to assist with modification tracking.
 */
package org.apache.jsp.jsp;

import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.jsp.*;

public final class Demo01_jsp extends org.apache.jasper.runtime.HttpJspBase
    implements org.apache.jasper.runtime.JspSourceDependent,
                 org.apache.jasper.runtime.JspSourceImports {

  private static final javax.servlet.jsp.JspFactory _jspxFactory =
          javax.servlet.jsp.JspFactory.getDefaultFactory();

  private static java.util.Map<java.lang.String,java.lang.Long> _jspx_dependants;

  private static final java.util.Set<java.lang.String> _jspx_imports_packages;

  private static final java.util.Set<java.lang.String> _jspx_imports_classes;

  static {
    _jspx_imports_packages = new java.util.HashSet<>();
    _jspx_imports_packages.add("javax.servlet");
    _jspx_imports_packages.add("javax.servlet.http");
    _jspx_imports_packages.add("javax.servlet.jsp");
    _jspx_imports_classes = null;
  }

  private volatile javax.el.ExpressionFactory _el_expressionfactory;
  private volatile org.apache.tomcat.InstanceManager _jsp_instancemanager;

  public java.util.Map<java.lang.String,java.lang.Long> getDependants() {
    return _jspx_dependants;
  }

  public java.util.Set<java.lang.String> getPackageImports() {
    return _jspx_imports_packages;
  }

  public java.util.Set<java.lang.String> getClassImports() {
    return _jspx_imports_classes;
  }

  public javax.el.ExpressionFactory _jsp_getExpressionFactory() {
    if (_el_expressionfactory == null) {
      synchronized (this) {
        if (_el_expressionfactory == null) {
          _el_expressionfactory = _jspxFactory.getJspApplicationContext(getServletConfig().getServletContext()).getExpressionFactory();
        }
      }
    }
    return _el_expressionfactory;
  }

  public org.apache.tomcat.InstanceManager _jsp_getInstanceManager() {
    if (_jsp_instancemanager == null) {
      synchronized (this) {
        if (_jsp_instancemanager == null) {
          _jsp_instancemanager = org.apache.jasper.runtime.InstanceManagerFactory.getInstanceManager(getServletConfig());
        }
      }
    }
    return _jsp_instancemanager;
  }

  public void _jspInit() {
  }

  public void _jspDestroy() {
  }

  public void _jspService(final javax.servlet.http.HttpServletRequest request, final javax.servlet.http.HttpServletResponse response)
      throws java.io.IOException, javax.servlet.ServletException {

    if (!javax.servlet.DispatcherType.ERROR.equals(request.getDispatcherType())) {
      final java.lang.String _jspx_method = request.getMethod();
      if ("OPTIONS".equals(_jspx_method)) {
        response.setHeader("Allow","GET, HEAD, POST, OPTIONS");
        return;
      }
      if (!"GET".equals(_jspx_method) && !"POST".equals(_jspx_method) && !"HEAD".equals(_jspx_method)) {
        response.setHeader("Allow","GET, HEAD, POST, OPTIONS");
        response.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, "JSP 只允许 GET、POST 或 HEAD。Jasper 还允许 OPTIONS");
        return;
      }
    }

    final javax.servlet.jsp.PageContext pageContext;
    javax.servlet.http.HttpSession session = null;
    final javax.servlet.ServletContext application;
    final javax.servlet.ServletConfig config;
    javax.servlet.jsp.JspWriter out = null;
    final java.lang.Object page = this;
    javax.servlet.jsp.JspWriter _jspx_out = null;
    javax.servlet.jsp.PageContext _jspx_page_context = null;


    try {
      response.setContentType("text/html; charset=UTF-8");
      pageContext = _jspxFactory.getPageContext(this, request, response,
      			null, true, 8192, true);
      _jspx_page_context = pageContext;
      application = pageContext.getServletContext();
      config = pageContext.getServletConfig();
      session = pageContext.getSession();
      out = pageContext.getOut();
      _jspx_out = out;

      out.write("\r\n");
      out.write("<!DOCTYPE html>\r\n");
      out.write("<html>\r\n");
      out.write("<head>\r\n");
      out.write("<meta http-equiv=\"Content-Type\" content=\"text/html; charset=UTF-8\">\r\n");
      out.write("<title>Insert title here</title>\r\n");
      out.write("</head>\r\n");
      out.write("<body>\r\n");
      out.write("\t<h1>Hello,bailu!</h1>\r\n");
      out.write("</body>\r\n");
      out.write("</html>");
    } catch (java.lang.Throwable t) {
      if (!(t instanceof javax.servlet.jsp.SkipPageException)){
        out = _jspx_out;
        if (out != null && out.getBufferSize() != 0)
          try {
            if (response.isCommitted()) {
              out.flush();
            } else {
              out.clearBuffer();
            }
          } catch (java.io.IOException e) {}
        if (_jspx_page_context != null) _jspx_page_context.handlePageException(t);
        else throw new ServletException(t);
      }
    } finally {
      _jspxFactory.releasePageContext(_jspx_page_context);
    }
  }
}

不想看全部代码的来看我这里的关键部分,具体如下图所示:

在这里插入图片描述
说明:我们可以看到,Java 通过out.write();方法将 JSP 标签输出,并对其他元素做了处理。

  1. 这也就是当初为什么出现 JSP 的原因,使用 JSP 比 Java 节省了大量的代码。
  2. 同时论证了 JSP 文件编译后首先生成的是 Servlet。
  3. 也就可以说,JSP 本质就是 Servlet,最终也是 Java 代码。

五、访问 JSP 文件的流程

到此,我们就得知,JSP 文件初次保存加载编译会先生成 Servlet,并进行之后的编译处理。所以,除去浏览器缓存的原因,初次访问 JSP 页面你会感到速度很慢,之后再访问就比较快了。

是否是第一次访问 JSP 文件的流程,具体如下图所示:

在这里插入图片描述
但是请注意:如果你的 JSP 文件进行了修改,再次点击保存发布会重新编译,又会重新走编译的流程。


文章作者白鹿第一帅作者主页https://blog.csdn.net/qq_22695001,未经授权,严禁转载,侵权必究!


总结

本文给大家介绍了 JSP 页面发布之后编译的流程,从 JSP 文件到 Sevlet(Java文件)再到 .class 文件最后到二进制机器码,剖析了为何 JSP 的本质即 Servlet,便于大家之后对 MVC 模式更进一步了解,加深对于 JSP 在架构中所处层次的掌握。还有建议大家养成一个好习惯:看源码!源码是一切!

在这里插入图片描述


我是白鹿,一个不懈奋斗的程序猿。望本文能对你有所裨益,欢迎大家的一键三连!若有其他问题、建议或者补充可以留言在文章下方,感谢大家的支持!

  • 15
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 7
    评论
### 回答1: JSP,即JavaServer Pages,是一种基于Java语言的动态网页技术。它可以实现在HTML页面中嵌入Java代码,并且与JavaBean、JSTL等技术一起来简化网页制作。 而Servlet,即Java Servlet,是Java语言实现的一种技术,也是Web开发的基础之一。它可以接收请求和响应数据,处理客户端和服务器之间的请求和响应。与JSP不同,ServletJava代码写在Servlet容器中,然后通过HTTP请求进行调用。 简单来说,JSP更多地用于视图层,Servlet更多地用于控制层。 ### 回答2: JSPJava Server Pages)是一种用于开发动态网页的Java技术。它将Java代码嵌入到HTML页面中,使得可以动态地生成网页内容。JSP页面在服务器端被解析和编译Java Servlet,然后由服务器执行,最终生成动态的HTML网页。 JSPServlet都属于Java Web开发的技术,但有一些区别: 1. 语法:JSP使用HTML作为主要的页面语言,通过<% %>标签将Java代码嵌入到HTML中。而Servlet是纯Java代码,没有HTML的限制。 2. 关注点:JSP主要关注于页面的视图层,用于展示动态内容。而Servlet主要关注于业务逻辑的处理,负责控制逻辑、数据库交互等后台操作。 3. 复用性:由于JSP中有大量的HTML和页面样式,因此JSP页面通常在视图层面上的复用性较差。而Servlet的复用性较好,可以通过返回不同的URL来处理不同的请求。 4. 可维护性:JSP的维护相对容易,因为它使用HTML作为主要语言,开发人员和设计人员可以分别处理页面和代码。而Servlet通常需要在Java代码中处理视图相关的逻辑,因此维护上的复杂度较高。 5. 异常处理:在JSP中,异常通常通过try-catch块捕获并显示错误页面。而Servlet可以使用Java异常处理机制,从而更好地控制异常情况和错误处理。 总的来说,JSPServlet的一种特殊形式,用于方便地开发和维护动态网页。JSP重点关注于视图层面的实现,而Servlet则更多地关注于实现后台的业务逻辑。 ### 回答3: JSPJava Server Pages)是一种用于构建动态Web页面的Java技术。使用JSP,可以在HTML页面中嵌入Java代码,并且这些代码在服务器端被执行,生成最终的HTML页面并返回给客户端浏览器。 JSPServlet都是Java在Web开发中的重要组成部分,但它们有一些区别: 1. 编程模型:JSP是基于HTML的,在HTML页面中嵌入Java代码。而Servlet是完全用Java编写的,没有HTML的限制。 2. 语法:JSP采用类似HTML的标记语法,通过使用<% %>包裹Java代码。而Servlet则是纯Java代码,没有特定的标记。 3. 执行方式:JSP在服务器启动时会被编译Servlet类,然后被服务器加载和执行。而Servlet直接通过Java编译编译成字节码文件,然后被服务器加载和执行。 4. 可读性与维护性:由于JSP采用HTML的标记语法,所以对于前端开发者来说更易于理解和维护。而Servlet则需要熟悉Java的开发人员来编写和维护。 5. 用途:JSP主要用于页面展示和交互,对于前端页面的动态生成非常适用。而Servlet则更加灵活,可以实现更复杂的业务逻辑和控制。 总的来说,JSP适合于前端页面的展示和交互,而Servlet适合于后端的业务逻辑处理。实际开发中,JSPServlet可以结合使用,共同完成一个完整的Web应用程序。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

白鹿第一帅

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值