当前位置: 首页 > 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.dtcms.com/a/360269.html

相关文章:

  • 血缘元数据采集开放标准:OpenLineage Integrations Compatibility Tests Structure
  • 一句话PHP木马——Web渗透测试中的隐形杀手
  • Java GC 销毁机制 与 Redis 过期策略深度对比
  • 微知-Mellanox OFED编译的一些细节?无法编译怎么办?如何添加自定义编译选项?
  • linux系统学习(14.日志管理)
  • Day 01(02): 精读HDFS概念
  • QML Chart组件之图例
  • leetcode_74 搜索二维矩阵
  • tiktok弹幕 X-Bogus
  • Self-Attention的实现
  • nginx-增加VTS模块
  • [光学原理与应用-357]:ZEMAX - 分析 - 光线迹点
  • Ubuntu 中复制粘贴快捷键
  • 中国 AI 应用出海研究:出海逻辑和注意事项
  • Dreamore AI-解读并描绘你的梦境
  • USBX移植(X是eXtended的意思)
  • 基于i.MX6ULL的RAM Disk驱动开发
  • Linux目录和命令介绍
  • 如何建立奢侈品牌的数字资产安全管控体系?
  • leetcode 371 两个整数之和
  • 智能数据建设与治理 Dataphin-数仓分层
  • 基于AI的大模型在S2B2C商城小程序中的应用与定价策略自我评估
  • “机器人管家“离我们还有多远?
  • 7.3 el-menu
  • 【C2000】C2000的硬件设计指导与几点意见
  • 官方 API 与网络爬虫的技术特性对比及选型分析
  • shell编程基础入门-2
  • Overleaf中文显示
  • 把 `QVector<QPointF>` 写入文件:三种风格、三段独立可编译的完整代码
  • APB协议​​ 构建一个完整的 ​​UVM验证VIP Agent介绍类的要素