zoukankan      html  css  js  c++  java
  • JSP 原理

    1、JSP的由来


      在JSP出现之前。为了实现动态网页的效果,server端利用 Servlet 的输出流向client发送HTML标签以及HTML页面中的内容,可是在多数动态网页中,绝大部分内容是静态的,仅仅有少量内容须要动态实现。可是为了这少量的动态内容。程序员依旧要用Servlet 输出当中全部的静态内容,这就使得整个Servlet 程序代码很臃肿,导致Servlet 的开发效率很低下。

      为了弥补Servlet 的缺陷,SUN公司在Servlet 的基础上推出了JSP(Java Server Pages)技术作为解决方式。JSP是简化Servlet 编写的一种技术,它由态部分和动态部分两部分组成。静态部分用于写入标准的HTML标签及内容。动态部分就是嵌入的Java代码与JSP动态标签了。通过这样的方式。使静态的部分直接使用HTML代码编写,对于动态的内容则使用 JAVA 脚本编写。

      对于Servlet 来说,不管动态、静态都用Java代码编写;而JSP则将静态的分出来,所实用HTML写(底层还是使用Java包装);动态的用Java 写。究其本质还是一样的。所以说。JSP的本质就是一种特殊的Servlet 。


    2、JSP的语法


       JSP = HTML + Java 脚本 + JSP 标签(指令),JSP中三种Java 脚本:

      ● <%...%>:Java代码片段,用于定义0~N条Java 语句,方法中可以写什么,这里面就能放什么;

      ● <%=  %>:Java 表达式,用于输出一条表达式或变量的结果。

     response.getWriter().print() 方法中可以写什么,这里面就行写什么;

      ● <%! … %> :声明,用来创建类的成员变量和成员方法,Java 类中可以写什么,这里面就行写什么,要注意的是,里面的内容不在 _jspService() 方法之内。直接被JSP转化后的类体包括。


    3、JSP 原理


      前面已经阐述过。JSP的本质实质是一种特殊形式的 Servlet :

      ● 当用户訪问一个JSP页面时。会向 Servlet 容器(这里是Tomcat)发出请求;

      ● 假设这个JSP页面是第一次被訪问或者这个页面被修改过时,server会把JSP 编译成 .java文件。当然,这个.java 就是一个servlet类,然后再把 .java 文件编译成.class 文件。

    由于编译会耗费一定时间,所以页面在第一次被訪问或修改后被訪问时会花费较长的訪问时间;

      ● 创建该类对象,最后由Servlet 容器调用它的service() 方法。

      ● 第二次请求同一JSP时,直接调用service() 方法。


    4、验证JSP原理


      首先,我们来写一个hello.jsp文件:

    <span style="font-size:18px;">
    <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
    <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
    
    <%
    String s = request.getHeader("User-Agent");
    %>
    
    <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
    <html>
      <head>
        <title>My JSP 'a.jsp' starting page</title>
        
    	<meta http-equiv="pragma" content="no-cache">
    	<meta http-equiv="cache-control" content="no-cache">
    	<meta http-equiv="expires" content="0">    
    	<meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
    	<meta http-equiv="description" content="This is my page">
    	<!--
    	<link rel="stylesheet" type="text/css" href="styles.css">
    	-->
    
      </head>
      
      <body>
    <table border="1" align="center" width="50%"> 
    	<tr>
    	  <th>姓名</th>
    	  <th>年龄</th>
    	  <th>性别</th>
    	</tr>
    <%
    	for(int i = 0; i < 10; i++) {
    %>	
    	<tr>
    	  <td>张三</td>
    	  <td>18</td>
    	  <td>男</td>
    	</tr>
    <%  }%>
    </table>
    
    
    
    <%!
    public void fun1() {
    	System.out.println("hello");
    }
    
    %>
    <%int a = 10; %>
    
    <%a++; %>
    
    <%=a %>
    
      </body>
    </html>
    
    </span>
      然后让我们在tomcat 下找到被编译成的.java 文件。为了节省空间。我把一些解释标记在代码凝视中:

    <span style="font-size:18px;">
    /*
     * Generated by the Jasper component of Apache Tomcat
     * Version: Apache Tomcat/7.0.42
     * Generated at: 2015-12-27 10:03:19 UTC
     * Note: The last modified time of this file was set to
     *       the last modified time of the source file after
     *       generation to assist with modification tracking.
     */
    package org.apache.jsp;
    
    import javax.servlet.*;
    import javax.servlet.http.*;
    import javax.servlet.jsp.*;
    import java.util.*;
    
    public final class hello_jsp extends org.apache.jasper.runtime.HttpJspBase
        implements org.apache.jasper.runtime.JspSourceDependent {
    
    
    public void fun1() {  //要特别注意,这是在<%! %>中定义的方法,没有被放在service方法中
    	System.out.println("hello");
    }
    
    
      private static final javax.servlet.jsp.JspFactory _jspxFactory =
              javax.servlet.jsp.JspFactory.getDefaultFactory();
    
      private static java.util.Map<java.lang.String,java.lang.Long> _jspx_dependants;
    
      private javax.el.ExpressionFactory _el_expressionfactory;
      private org.apache.tomcat.InstanceManager _jsp_instancemanager;
    
      public java.util.Map<java.lang.String,java.lang.Long> getDependants() {
        return _jspx_dependants;
      }
    
      public void _jspInit() {
        _el_expressionfactory = _jspxFactory.getJspApplicationContext(getServletConfig().getServletContext()).getExpressionFactory();
        _jsp_instancemanager = org.apache.jasper.runtime.InstanceManagerFactory.getInstanceManager(getServletConfig());
      }
    
      public void _jspDestroy() {
      }
    
      public void _jspService(final javax.servlet.http.HttpServletRequest request, final javax.servlet.http.HttpServletResponse response)
            throws java.io.IOException, javax.servlet.ServletException { //这是_jspService()方法
    
        final javax.servlet.jsp.PageContext pageContext; //内置对象的初始化
        javax.servlet.http.HttpSession session = null;
        final javax.servlet.ServletContext application;
        final javax.servlet.ServletConfig config;
        javax.servlet.jsp.JspWriter out = null;
        final java.lang.Object page = this;
        javax.servlet.jsp.JspWriter _jspx_out = null;
        javax.servlet.jsp.PageContext _jspx_page_context = null;
    
    
        try {
          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;
    
          out.write("
    ");
          out.write("
    ");
          out.write("
    ");
    
    String s = request.getHeader("User-Agent");  //它的原身是<span style="font-family: Arial, Helvetica, sans-serif;">String s = request.getHeader("User-Agent");。被直接拿过来了</span>
    
    
          out.write("
    ");
          out.write("
    ");
          out.write("<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
    ");
          out.write("<html>
    ");   //这些都是HTML代码。底层被包装成和Servlet 一样的实现方式
          out.write("  <head>
    ");
          out.write("    <title>My JSP 'a.jsp' starting page</title>
    ");
          out.write("    
    ");
          out.write("	<meta http-equiv="pragma" content="no-cache">
    ");
          out.write("	<meta http-equiv="cache-control" content="no-cache">
    ");
          out.write("	<meta http-equiv="expires" content="0">    
    ");
          out.write("	<meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
    ");
          out.write("	<meta http-equiv="description" content="This is my page">
    ");
          out.write("	<!--
    ");
          out.write("	<link rel="stylesheet" type="text/css" href="styles.css">
    ");
          out.write("	-->
    ");
          out.write("
    ");
          out.write("  </head>
    ");
          out.write("  
    ");
          out.write("  <body>
    ");
          out.write("<table border="1" align="center" width="50%"> 
    ");
          out.write("	<tr>
    ");
          out.write("	  <th>姓名</th>
    ");
          out.write("	  <th>年龄</th>
    ");
          out.write("	  <th>性别</th>
    ");
          out.write("	</tr>
    ");
    
    	for(int i = 0; i < 10; i++) {  //这也是直接拿过来的
    
          out.write("	
    ");
          out.write("	<tr>
    ");
          out.write("	  <td>张三</td>
    ");
          out.write("	  <td>18</td>
    ");
          out.write("	  <td>男</td>
    ");
          out.write("	</tr>
    ");
      }
          out.write("
    ");
          out.write("</table>
    ");
          out.write("
    ");
          out.write("
    ");
          out.write("
    ");
          out.write('
    ');
          out.write('
    ');
    int a = 10;   //这些是定义的变量,能够看到是放在了service方法中的
          out.write("
    ");
          out.write("
    ");
    a++; 
          out.write("
    ");
          out.write("
    ");
          out.print(a );
          out.write("
    ");
          out.write("
    ");
          out.write("  </body>
    ");
          out.write("</html>
    ");
        } catch (java.lang.Throwable t) {
          if (!(t instanceof javax.servlet.jsp.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);
            else throw new ServletException(t);
          }
        } finally {
          _jspxFactory.releasePageContext(_jspx_page_context);
        }
      }
    }
    
    </span>
      得出结论,JSP中尽管能直接写出HTML代码,可是在底层依旧是被包装成 Servlet 实现方式的。所以更印证了JSP是特殊的Servlet 。

      其次。在<%  %> 和 <%= %> 脚本中定义的Java 代码都会放在JSP 的 _jspService() 方法中(实际上就是Servlet中的service 方法)。而<%! %> 脚本中定义的却会放到 hello_jsp 类的成员位置的,这一点非常重要,由于JSP中鼎鼎大名的九大内置对象是在_jspService() 方法中初始化的,仅仅有在本方法中才可以使用内置对象。所以<%! %> 脚本中是不能使用内置对象的(在实际开发中,本脚本非常少用到)。


    5、JSP和HTML、Servlet 的比較


      ● Servlet:

        缺点:不适合设置html响应体。须要大量的response.getWriter().print("<html>")

        长处:动态资源,能够编程。

      ● HTML:

        缺点:html是静态页面。不能包括动态信息

        长处:不用为输出html标签而发愁

      ● JSP:

        长处:在原有html的基础上加入java脚本。构成jsp页面。


    6、JSP 和 Servlet 的分工


      在设计中,JSP和Servlet 是相互配合使用的。其分工为:

      ● JSP:

        作为请求发起页面。比如显示表单、超链接,并将请求发给 Servlet 。

        作为请求结束页面,比如显示数据。

      ● Servlet:

        作为请求中处理数据的环节。


      小结:JSP页面必需要在JSPserver内执行。如tomcat weblogic,jboss(这些都是 apache 中的子项目,apache是 Web 应用server,而 tomcat 等能够说是JSP或 Servle 的 Web 容器。简称Servlet 容器)等;JSP页面的訪问者无须安装不论什么client,也不需要执行Java 环境,由于JSP页面输送到client的是标准HTML页面。




  • 相关阅读:
    dns解析后ping的居然不是自己的ip
    Ubuntu修改默认使用的bash
    安装 libbpg
    libnccl安装
    安装opencv
    tcpdump使用
    jQuery类操作
    jQuery对象和DOM对象的相互转换
    jQuery入口函数
    什么是外边距重叠?重叠的结果是什么?
  • 原文地址:https://www.cnblogs.com/jzssuanfa/p/6971440.html
Copyright © 2011-2022 走看看