6.1.4 分析由JSP生成的Servlet代码

优质
小牛编辑
132浏览
2023-12-01

由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节详细介绍。