当前位置: 首页 > news >正文

JSP 原理深度解析

JSP 原理深度解析

一、JSP 本质与工作原理

1. JSP 的本质

JSP (JavaServer Pages) 的本质是 Servlet 的变体。JSP 页面在第一次被访问时,会被 Web 容器(如 Tomcat)翻译和编译成 Servlet 类,然后执行。

浏览器请求 JSP 页面
JSP 是否已编译?
容器翻译 JSP 为 Java Servlet 代码
编译 Java 代码生成 .class 文件
实例化 Servlet 对象
调用 _jspService 方法
生成响应内容
返回 HTML 响应给浏览器

2. JSP 到 Servlet 的转换过程

示例 JSP 代码 (hello.jsp):

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head><title>Hello</title>
</head>
<body><h1>Hello, <%= request.getParameter("name") %>!</h1><p>Current time: <%= new java.util.Date() %></p>
</body>
</html>

转换后的 Servlet 代码片段:

public final class hello_jsp extends HttpJspBase {public void _jspService(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException {// 初始化工作PageContext pageContext = null;HttpSession session = null;ServletContext application = null;ServletConfig config = null;JspWriter out = null;Object page = this;// 设置内容类型response.setContentType("text/html;charset=UTF-8");try {// 获取输出流out = pageContext.getOut();// 输出HTML内容out.write("<html>\n");out.write("<head>\n");out.write("    <title>Hello</title>\n");out.write("</head>\n");out.write("<body>\n");out.write("    <h1>Hello, ");// 嵌入的Java代码out.print(request.getParameter("name"));out.write("!</h1>\n");out.write("    <p>Current time: ");// 嵌入的Java代码out.print(new java.util.Date());out.write("</p>\n");out.write("</body>\n");out.write("</html>");} catch (Exception e) {// 异常处理} finally {// 清理资源}}
}

二、JSP 核心组件解析

1. JSP 指令 (Directives)

<%@ page %>        <!-- 页面级别设置 -->
<%@ include %>     <!-- 包含其他文件 -->
<%@ taglib %>      <!-- 引入标签库 -->

page 指令详解:

<%@ page language="java" contentType="text/html; charset=UTF-8"pageEncoding="UTF-8"import="java.util.*, java.text.*"session="true"buffer="8kb"autoFlush="true"isThreadSafe="true"info="页面描述信息"errorPage="/error.jsp"isErrorPage="false"isELIgnored="false"deferredSyntaxAllowedAsLiteral="false"trimDirectiveWhitespaces="true"
%>

2. JSP 脚本元素

元素类型语法转换结果示例
脚本片段<% code %>直接插入到 _jspService 方法中<% int count = 0; %>
表达式<%= expression %>转换为 out.print()<%= user.getName() %>
声明<%! declaration %>添加到 Servlet 类级别<%! private int instanceVar; %>

3. JSP 隐含对象

JSP 提供 9 个无需声明即可使用的隐含对象:

对象类型作用域说明
requestHttpServletRequestrequest客户端请求信息
responseHttpServletResponsepage服务器响应信息
outJspWriterpage输出流对象
sessionHttpSessionsession用户会话对象
applicationServletContextapplication应用上下文
configServletConfigpageServlet 配置
pageContextPageContextpage页面上下文
pageObjectpage当前页面实例
exceptionThrowablepage异常对象(仅errorPage)

三、JSP 生命周期深度解析

1. 翻译阶段 (Translation Phase)

容器将 JSP 文件解析为 Java Servlet 源代码:

  • 解析指令和脚本元素
  • 验证语法正确性
  • 生成对应的 Java 代码

2. 编译阶段 (Compilation Phase)

  • 使用 JDK 编译生成的 Java 代码
  • 生成 .class 字节码文件
  • 通常存放在容器的 work 目录中

Tomcat 中的存放路径示例:

${CATALINA_BASE}/work/Catalina/localhost/${APP_NAME}/org/apache/jsp/

3. 初始化阶段 (Initialization Phase)

// 生成的Servlet继承HttpJspBase
public class hello_jsp extends HttpJspBase {// 初始化方法public void jspInit() {// 对应JSP中的 <%! %> 声明初始化}
}

4. 执行阶段 (Execution Phase)

每次请求时调用 _jspService() 方法:

  1. 初始化隐含对象
  2. 执行脚本片段和表达式
  3. 生成响应内容

5. 销毁阶段 (Destruction Phase)

public void jspDestroy() {// 清理资源// 对应JSP中的 <%! %> 声明销毁逻辑
}

四、JSP 高级特性

1. 自定义标签库原理

<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<c:forEach items="${users}" var="user"><p>${user.name}</p>
</c:forEach>

标签处理类:

public class ForEachTag extends SimpleTagSupport {private Collection<?> items;private String var;public void setItems(Collection<?> items) {this.items = items;}public void setVar(String var) {this.var = var;}@Overridepublic void doTag() throws JspException, IOException {if (items != null) {for (Object item : items) {getJspContext().setAttribute(var, item);getJspBody().invoke(null);}}}
}

2. EL 表达式 (Expression Language) 原理

${user.profile.address.city}

转换过程:

// 上述EL表达式被转换为:
pageContext.findAttribute("user").getProfile().getAddress().getCity();

3. JSTL (JSP Standard Tag Library) 工作原理

<c:set var="message" value="Hello World" />
<fmt:formatDate value="${now}" pattern="yyyy-MM-dd" />

五、性能优化与最佳实践

1. 预编译 JSP

# 使用Tomcat的JSP预编译工具
javac -cp ${TOMCAT_HOME}/lib/* org/apache/jsp/*.java

2. 合理使用包含机制

<%-- 静态包含:编译时包含 --%>
<%@ include file="/WEB-INF/header.jsp" %><%-- 动态包含:运行时包含 --%>
<jsp:include page="/WEB-INF/sidebar.jsp" />

3. 避免常见的性能问题

不佳实践:

<%-- 在循环内创建对象 --%>
<% for (int i = 0; i < 1000; i++) { %><%= new java.util.Date() %>  <!-- 每次循环创建新对象 -->
<% } %>

优化实践:

<%-- 在循环外创建对象 --%>
<% java.util.Date now = new java.util.Date(); %>
<% for (int i = 0; i < 1000; i++) { %><%= now %>  <!-- 复用同一对象 -->
<% } %>

六、JSP 与现代技术的结合

1. 与 Spring MVC 集成

<%@ taglib prefix="spring" uri="http://www.springframework.org/tags" %>
<%@ taglib prefix="form" uri="http://www.springframework.org/tags/form" %><form:form modelAttribute="user"><form:input path="name" /><form:errors path="name" />
</form:form>

2. AJAX 与 JSP 结合

<script>
function loadUserInfo() {$.get('${pageContext.request.contextPath}/user/ajax', function(data) {$('#userInfo').html(data);});
}
</script>

七、调试与故障排除

1. 查看生成的 Servlet 代码

web.xml 中配置开发模式:

<servlet><servlet-name>jsp</servlet-name><servlet-class>org.apache.jasper.servlet.JspServlet</servlet-class><init-param><param-name>keepgenerated</param-name><param-value>true</param-value></init-param><init-param><param-name>development</param-name><param-value>true</param-value></init-param>
</servlet>

2. 常见错误分析

  • 编译错误: 检查 JSP 语法和 Java 代码
  • 运行时错误: 查看服务器日志中的异常堆栈
  • 内存泄漏: 检查是否在声明中使用了实例变量

总结

JSP 技术虽然逐渐被现代前端框架取代,但理解其底层原理对于深入掌握 Java Web 开发至关重要。JSP 的核心价值在于:

  1. 分离表现与逻辑: 通过标签和 EL 表达式减少 Java 代码在页面中的出现
  2. 组件化开发: 通过自定义标签和 JSTL 实现可重用组件
  3. 高效的开发模式: 修改后无需重新编译整个应用

尽管现在推荐使用模板引擎(Thymeleaf、FreeMarker)或前后端分离架构,但 JSP 的原理仍然是理解 Web 视图技术的基础。

http://www.xdnf.cn/news/1412173.html

相关文章:

  • MATLAB R2010b系统环境(四)MATLAB帮助系统
  • 【GPT入门】第62课 情感对话场景模型选型、训练与评测方法,整体架构设计
  • 深度学习篇---MobileNet网络结构
  • 五分钟聊一聊AQS源码
  • globals() 小技巧
  • 仅有一张Fig的8分文章 胞外囊泡lncRNA+ CT 多模态融合模型,AUC 最高达 94.8%
  • 【LeetCode修行之路】算法的时间和空间复杂度分析
  • 大数据毕业设计选题推荐-基于大数据的大气和海洋动力学数据分析与可视化系统-Spark-Hadoop-Bigdata
  • ESP32C3 系列实战(1) --点亮小灯
  • Wi-Fi技术——物理层技术
  • 使用Cadence工具完成数模混合设计流程简介
  • LangChain核心抽象:Runnable接口深度解析
  • leetcode_48 旋转图像
  • FFMPEG学习任务
  • 第 14 篇:K-Means与聚类思维——当AI在没有“标准答案”的世界里寻宝
  • 【C2000】C2000的硬件设计指导与几点意见
  • 开源知识抽取框架 推荐
  • 京东获取商品评论指南,实时关注用户反馈
  • 官方 API 与网络爬虫的技术特性对比及选型分析
  • Unity学习----【数据持久化】二进制存储(三)--文件夹操作
  • OpenStack 01:介绍
  • 暄桐林曦老师关于静坐常见问题的QA
  • 基于GA遗传优化的双向LSTM融合多头注意力(BiLSTM-MATT)时间序列预测算法matlab仿真
  • windows系统中的docker,xinference直接运行在容器目录和持载在宿主机目录中的区别
  • isat将标签转化为labelme格式后,labelme打不开的解决方案
  • MyBatis 黑马 辅助配置,数据库连接池
  • 柔性数组与不定长数据
  • 【秋招笔试】2025.08.31饿了么秋招笔试题
  • SPMTE 2022概述
  • 线程池常见面试问答