6.1.4 分析由JSP生成的Servlet代码
由6.1.2节的simple.jsp文件生成的simple_jsp.java的源代码如下:
package org.apache.jsp;
import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.jsp.*;
// 由<%@page import... %>引用的Java类或包将被作为import语句插入到Servlet类中
import java.text.SimpleDateFormat;
import java.util.Date;
public final class simple_jsp extends org.apache.jasper.runtime.HttpJspBase
implements org.apache.jasper.runtime.JspSourceDependent
{
private static final JspFactory _jspxFactory = JspFactory.getDefaultFactory();
private static java.util.List _jspx_dependants;
private javax.el.ExpressionFactory _el_expressionfactory;
private org.apache.AnnotationProcessor _jsp_annotationprocessor;
public Object getDependants()
{
return _jspx_dependants;
}
public void _jspInit()
{
_el_expressionfactory =
_jspxFactory.getJspApplicationContext(getServletConfig().getServletContext()).getExpressionFactory();
_jsp_annotationprocessor = (org.apache.AnnotationProcessor)
getServletConfig().getServletContext().getAttribute(org.apache.AnnotationProcessor.class.getName());
}
public void _jspDestroy()
{
}
public void _jspService(HttpServletRequest request, HttpServletResponse response)
throws java.io.IOException, ServletException
{
// 定义JSP中的内置对象
PageContext pageContext = null;
HttpSession session = null;
ServletContext application = null;
ServletConfig config = null;
JspWriter out = null;
Object page = this;
JspWriter _jspx_out = null;
PageContext _jspx_page_context = null;
try
{
// <%@ page contentType="text/html; charset=UTF-8" ... %>
// 被翻译成调用setContentType方法设置Content-Type字段
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;
// JSP页面中静态的部分直接使用write方法输出到客户端
out.write("\r\n");
out.write("\r\n");
out.write("<html>\r\n");
out.write("<head>\r\n");
out.write("\t<title>简单的JSP程序</title>\r\n");
out.write("</head>\r\n");
out.write("<body>\r\n");
out.write(" 当前日期和时间:\r\n");
out.write(" ");
// 写在<% ... %>中的Java代码被直接插入到Servlet类中
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
out.println(dateFormat.format(new Date()));
out.write("\r\n");
out.write("<p/>\r\n");
out.write("name请求参数值:");
// 写在<%= ... %>中的Java代码被翻译成使用print方法输出,
// 也就是说,<%= ... %>中的Java代码实际上是一个返回值的表达式
out.print( request.getParameter("name") );
out.write("\r\n");
out.write("</body>\r\n");
out.write("</html> ");
}
catch (Throwable t)
{
if (!(t instanceof SkipPageException))
{
out = _jspx_out;
if (out != null && out.getBufferSize() != 0)
try { out.clearBuffer(); } catch (java.io.IOException e) {}
if (_jspx_page_context != null) _jspx_page_context.handlePageException(t);
}
} finally {
_jspxFactory.releasePageContext(_jspx_page_context);
}
}
}
从上面的代码中可以得出以下结论:
1. 由JSP生成的Servlet类从org.apache.jasper.runtime.HttpJspBase类继承,HttpJspBase是HttpServlet类的子类,因此,从HttpJspBase继承的类也是Servlet类。HttpJspBase类除了继承HttpServet外,还实现了HttpJspPage接口,在HttpJspPage接口中定义了一个_jspService方法,HttpJspBase类中并未实现_jspService方法,因此,由JSP生成的Servlet类必须实现_jspService方法。_jspService方法相当于Servlet类中的service方法。从simple_jsp类的代码可以看出,在_jspService方法中加入了大量的代码,这些代码有很多都是由JSP页面代码翻译而来。
2. 由<%@page import... %>引用的Java包和类被翻译成Java的import语句。
3. JSP页面中使用<%@ page contentType="text/html; charset=UTF-8" ... %>设置Content-Type字段的语句被翻译成response.setContentType(...)方法的调用。
4. <% ... %>中的Java代码被直接插入到_jspService方法中。
5. <%= ... %>中的Java代码被当作一个值,并使用out.println(...)方法输出该值。
6. JSP页面中的静态部分被直接使用out.write(...)方法输出到客户端。
7. JSP页面中涉及到的9个内置对象中的6个在_jspService方法的开始位置被定义,并且在_jspSevice方法中初始化了这些内置对象。另外三个内置对象中的response和request就是_jspService方法的两个参数,另外一个exception对象必须将page指令的isErrorPage属性值设为true时才会创建,如下面代码所示:
<%@ page language="java" contentType="text/html; charset=UTF-8" isErrorPage="true" pageEncoding="UTF-8"%>
如果在JSP页面中使用上面的page指令,当JSP页面被翻译成Servlet类时,就会生成定义exception对象的Java代码,如下面代码所示:
Throwable exception =org.apache.jasper.runtime.JspRuntimeLibrary.getThrowable(request);
if (exception != null)
{
response.setStatus(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);
}
关于JSP的内置对象将在6.3节详细介绍。