JSP页面翻译原理

ZDNet软件频道 时间:2008-06-30 作者: | 中国IT实验室 我要评论()
本文关键词:原理 翻译 JSP
众所周知,每一个JSP页面都会被Web容器编译成一个Java类,供web容器调用,并且生成HTML叶面回馈给用户。

众所周知,每一个JSP页面都会被Web容器编译成一个Java类,供web容器调用,并且生成HTML叶面回馈给用户。而了解其中的变异方法和规则,对我们学习JSP是非常有好处的,可以说学习好了这个编译原理,就已经学习好了大部分的JSP知识,剩下的工作就只剩下熟记一些tablib和反复应用以使自己更加熟练而已了。。


先来看一下JSP页面所对应的Class的基本结构。每一个JSP页面都会被编译成成如下的格式样子,先给一个大致的印象,详细的说明在后面。

public class My$JSP extends HttpJSPBase {    static {}    public date$JSP() {}    private static boolean _JSPx_inited = false;    public final void _JSPx_init()      throws org.apache.jasper.runtime.JSPException {};    public void _JSP pageservice(HttpServletRequest request,                            HttpServletResponse response)      throws java.io.IOException, ServletException {      JSPFactory _JSPxFactory = null;      PageContext pageContext = null;      HttpSession session = null;      ServletContext application = null;      ServletConfig config = null;      JSPWriter out = null;      Object page = this;      String _value = null;      try {        if (_JSPx_inited == false) {          synchronized (this) {            if (_JSPx_inited == false) {                _JSPx_init();                _JSPx_inited = true;          }      }     }     _JSPxFactory = JSPFactory.getDefaultFactory();     response.setContentType("text/html;charset=ISO-8859-1");     pageContext = _JSPxFactory.getPageContext(this, request, response,                                              "", true, 8192, true);      application = pageContext.getServletContext();       config = pageContext.getServletConfig();       session = pageContext.getSession();       out = pageContext.getOut();      // HTML // begin [file="/date.JSP";from=(0,0);to=(7,6)]        out.write("rnrnrn" +                  "rnrnrn" +                  "The date isrn");        // end        // begin [file="/date.JSP";from=(7,8);to=(7,57)]        out.println((new java.util.Date()).toString());        // end        // HTML // begin [file="/date.JSP";from=(7,59);to=(10,7)]        out.write("rn    rn rn");        // end        } catch (Throwable t) {          if (out != null && out.getBufferSize() != 0) {            out.clearBuffer();          }          if (pageContext != null) {            pageContext.handlePageException(t);          }        } finally {           if (_JSPxFactory != null) {             _JSPxFactory.releasePageContext(pageContext);           }        }      }   }我们可以清楚地看到,这里面最重要的函数就是pageservice,web容器在编译好一个JSP类以后,
就申请这个类的对象,并且直接调用pageservice来获得Response,最后返回给客户。
作为细节,我们可以总结如下:

所有的JSP页面翻译出来的class,都从HttpJSPBase继承,并且命名为PageName$JSP 在第一次调用pageservice函数的时候,该class会进行一次初始化,而这个初始化函数是_JSPx_init,如果我们想,我们还可以自定义这个函数,来实现JSP页面的初始化。 这样的代码被转换成什么了?
这样的代码被直接转成Java代码放到pageservice函数里面。 这样的代码被转换成什么了?
这样的代码被翻译成成员函数和成员变量,也就是说,这些声明在JSP的生命周期内都是存在的。 HTML代码呢?
html代码直接被写到PrintWriter里面回馈给用户。非常的直接 为什么JSP页面有那么多省写方式,比如说session , out , page , context之类。
这都是在pageservice里面定义的临时变量,具体的初始化可以参看上面的例子代码,每一次调用JSP页面,这些变量都会被重新初始化一次。当然我们也可以方便的声明自己的变量。 省写方式 这么理解? 这种省写方式调用doSomething所得到的Object的toString(),然后直接写到out里面。相当于:

out.print(object.doSomethiing().toString())JavaBean 里面的scope定义了作用域范围,这个范围在这里的意思是?
这是Bean对象句柄保存的地方的意思。我们可以想象一下,一个page范围的Bean只是pageservice里面的一个局部变量,当一次处理结束后,这个变量就会被Java虚拟机回收。而session变量。而request级别的Bean就应该是JSP页面的成员变量。而session和application则不能在JSP页面class里面保存,而应该保存在JSP页面的调用对象里面。 关于命令,这个就太简单了,只是一个一个的对应为response.setContentType()的语句而已。 关于JSP页面转向问题。这个语句被翻译为getServletContext().getRequestDispatcher("/List.JSP").forward(req, res);语句。 遇到这个语句,JSP翻译器就会把这个文件的代码和现在文件的代码混合然后一起编译,生成JSP类。这个方法很好,可以让我们统一文档的样式,比如说吧header写成一个文件,,而把footer也写成一个JSP ,并且在index.html里面把这两个文件包含近来,这样,不管Content怎么变,上下样式都不会变,有利于样式的统一。


百度大联盟认证黄金会员Copyright© 1997- CNET Networks 版权所有。 ZDNet 是CNET Networks公司注册服务商标。
中华人民共和国电信与信息服务业务经营许可证编号:京ICP证010391号 京ICP备09041801号-159
京公网安备:1101082134