文章目录

  • 前言
  • 一、JSP 文件编译流程原理
  • 二、创建并运行待测试 JSP 页面
  • 三、查找 JSP 编译文件输出位置
    • 3.1、打开动态项目运行配置
    • 3.2、查看 JSP 编译文件输出位置
    • 3.3、查看 JSP 编译输出文件
  • 四、JSP 编译输出 Servlet 的论证
  • 五、访问 JSP 文件的流程
  • 总结

前言

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


一、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.HttpJspBaseimplements 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 文件进行了修改,再次点击保存发布会重新编译,又会重新走编译的流程。


总结

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


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

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

  1. MVC 模式/Servlet/JSP 编译原理剖析:Servlet 组件到底属于 MVC 模式的哪一层?

    文章目录 前言 一.回忆什么是 MVC 模式? 1.1.Model.View.Controller 组件介绍 1.2.明确 View 与 Controller 组件区别 二.什么是 Servlet? ...

  2. 编译原理实验c语言cfg文法,编译原理

    地址在符号表中引入指针previous,来连接上一个符号的首地址运行时存储空间组织活动记录用于管理函数变量的信息栈式存储过程进入和返回通过变更top和sp指针,实现活动记录的栈式处理静态链实现局部变量 ...

  3. JSP和Servlet是什么关系,JSP工作原理?

    JSP和Servlet是什么关系 其实这个问题在上面已经阐述过了,Servlet是一个特殊的Java程序,它运行于服务器的JVM中,能够依靠服务器的支持向浏览器提供显示内容.JSP本质上是Servle ...

  4. JSP编译成Servlet(四)JSP与Java行关系映射

    我们知道java虚拟机只认识class文件,要在虚拟机上运行就必须要遵守class文件格式,所以JSP编译成servlet后还需要进一步编译成class文件,但从JSP文件到java文件再到class ...

  5. jsp执行原理(详解)

    1. jsp的工作模式 jsp的工作模式是请求/相应模式,客户端首先发出HTTP请求,jsp程序收到请求后会进行处理并返回处理结果.在一个jsp文件第一次被请求时,jsp引擎(容器)把该jsp文件转换 ...

  6. JSP运行原理 九大内置对象 四大作用域

    Jsp的简介 jsp的全称是java server pages 意思为java的服务页面 jsp必须运行在服务器,不能脱离服务器单独运行 jsp是可以写java代码的html jsp的本质就是一个se ...

  7. JSP运行原理及运行过程

    学习J2EE时,初步了解了JSP,写过一篇小的总结:J2EE-JSP,当时还不是很了解它的运行机制. 学习DRP时,再次接触了JSP,在原来基础上有了更深的了解,了解了它的执行原理 1,JSP的运行原 ...

  8. jsp是什么以及jsp运行原理

    jsp是什么以及jsp运行原理 一.JSP的简单介绍 1. JSP(Java Server Pages)是由Sun Microsystems公司倡导.许多公司参与一起建立的一种动态网页技术标准.JSP ...

  9. java与servlet JSP_java面试精品全集[jsp与servlet部分]

    一.Jsp方面 1.forward 和redirect的区别 答:forward是服务器请求资源,服务器直接访问目标地址的URL,把那个URL的响应内容读取过来,然后把这些内容再发给浏览器,浏览器根本 ...

最新文章

  1. Wake-On-LAN待机或休眠模式中唤醒
  2. 天命剑之天命的含义--天行有悖,乃命羲和。
  3. SylixOS iMX6平台I2C总线驱动
  4. 复用的具体实现方式组合与继承的特点
  5. jquery对Select的操作
  6. android 获取apk中的素材图片
  7. 【Android 修炼手册】常用技术篇 -- Android 自定义 View
  8. django用户认证系统——登录4
  9. html5音乐播放器设计论文,基于微信小程序的音乐播放器设计和毕业论文
  10. 内存条上面参数详解_价格极低的国产颗粒内存条:光威弈系列Pro评测,超频表现超稳定...
  11. 风控成长之路---必知必会工具与业务
  12. linux 授信_Linux 配置双机SSH信任
  13. 企业到底需要何种数据分析挖掘工具?
  14. 中国行政区划 地级 js
  15. 大数据杀熟 算法_大数据“杀熟” 怎能让真正的刀俎逍遥法外
  16. java.util.concurrent 同步器框架详解
  17. 浅谈领导力理解和体会
  18. 支付业务与技术架构学习总结(7)——从金融牌照(三方支付牌照),看互联网巨头的金融布局
  19. 加强银行余额调节表在财务管理中的作用
  20. 瑞吉外卖-全网最全笔记-Day06

热门文章

  1. mapreduce复制连接的代码_我的 Hive 为什么跑不起来/跑得慢?看看是不是少了这几行代码?...
  2. 命题公式的主合取范式C语言,命题公式主范式的自动生成与形式输出.pdf
  3. c语言队列作用,队列(C语言)
  4. vue 报错error: ‘to‘ is defined but never used (no-unused-vars)
  5. IDEA 断点调试高级玩法 |debug高手必看!
  6. 导入Oracle 数据库镜像,创建Oracle虚拟机_01
  7. Redis-6.2.5 安装 Linux环境(单机)
  8. Github上Fork部署应用程序
  9. SpringBoot精选项目
  10. Java-instanceof和类型转换