zoukankan      html  css  js  c++  java
  • JSP基础知识

    http://topmanopensource.iteye.com/blog/518126

    JSP基础知识

    JSP本质上就是把Java代码嵌套到HTML中,然后经过JSP容器(Tomcat、Resin、Weblogic等)的编译执行,再根据这些动态代码的运行结果生成对应的HTML代码,从而可以在客户端的浏览器中正常显示。

    JSP运行原理

          如果JSP页面是第一次被请求运行,服务器的JSP编译器会生成JSP页面对应的JAVA代码,并且编译成类文件。当服务器再次收到对这个JSP页面请求的时候,会判断这个JSP页面是否被修改过,如果被修改过就会重新生成Java代码并且重新编译,而且服务器中的垃圾回收方法会把没用的类文件删除。如果没有修改过,服务器就会直接调用以前已经编译过的类文件。

    需要注意:

          是只有被请求过的页面才能生成对应的Java文件,没有请求的页面会在第一次请求的时候生成Java文件,当JSP页面被修改后,再次对这个页面进行请求才会重新生成对应的Java文件。

    JSP的优势

         JSP就是在HTML中嵌入Java代码,所以在本质上JSP程序就是Java程序,JSP程序继承了Java的一切优点。JSP程序有严格的Java语法和丰富的Java类库支持。

              JSP页面在服务器中都会被JSP编译器编译成对应的Servlet,所以就拥有Java跨平台的优点,所有的JSP程序,无需改动就可以方便的迁移到其他操作系统平台,这就是在其他动态脚本中所无法想象的。

    JSP脚本元素

    Jsp声明语法:<%! %>
            jsp声明只在当前页面有效,并且jsp声明中不能使用jsp内置对象。在其内声明的变量或方法为实例方法或实例变量

    Jsp中Java代码块语法:<% %>
          jsp中嵌入的java代码块称Scriptlet,默认为与jsp对应的Servlet类的service()方法中的代码块。Scriptlet位置由page指令中的method确定method值可以为service(),doGet(),doPost()等。在其内声明的变量为局部变量

    Jsp中的表达式语法:<%= %>用于在页面上输出表达式或变量的值


    JSP内置对象
     对象          类型
     request      javax.servlet.HttpServletRequest
     response     javax.servlet.HttpServletResponse
     pageContext    javax.servlet.jsp.PageContext
     application     javax.servlet.ServletContext
     out          javax.servlet.jsp.JspWriter
     config          javax.servlet.ServletConfig
     page          java.lang.Object(类似于java中的this关键字)
     session      javax.servlet.HttpSession
     exception     java.lang.Exception
       

    JSP的异常处理:
     Jsp和Java一样在运行时也有可能抛出异常,在发生异常的页面中可以添加如下代码
     <%@ page errorPage="errorpage.jsp" %>
     当前页面发生异常跳转至errorpage.jsp进行异常处理

     在errorpage.jsp页面中添加如下语句声明此页面是处理异常的页面
     <%@ page isErrorPage="true" %>

     在处理异常的页面可以直接访问exception内置对象,获取异常信息
     <% exception.printStackTrace(new PrintWriter(out)); %>


    JSP请求转发:
         request.getRequestDispatcher("success.jsp").forward(request,response);
     特点:

           地址栏不改变,我们称为请求转发,发起的是同一次请求,数据未丢失

     response.sendRedirect("success.jsp");
     特点:

           地址栏改变,我们称为重定向,发起的是两次请求,数据丢失


     
    中文乱码解决方式:
       post提交方式:通过request.setCharacterEncoding("utf-8");
             get提交(url传参)方式:String str= new String(request.getParameter("uName").getBytes("iso-8859-1"),"utf-8");


    关于JSP预编译:

          通过特殊的请求参数jsp_precomplie,它的值可以true或false;请求为ture那么Servlet容器仅仅对客户端请求的JSP文件进行预编译,即将JSP文件转化为Servlet类,但不运行Servlet;

    1.http://localhost:8080/helloapp/hello.jsp?jsp_precomplie=true
    2.找到Tomcat目录下work目录对应项目helloapp下的hello_jsp.class文件
    将其复制到WEB-INF/classes对应的子目录中
    3.在web.xml文件中配制hello_jsp类
     <servlet>
      <servlet-name>hi</servlet-name>
      <servlet-class>org.apache.jsp.hello_jsp</servlet-name>
     </servlet>

     <servlet-mapping>
      <servlet-name>hi</servlet-name>
      <url-pattern>/hello</url-pattern>
     </servlet-mapping> 

    以下为Eclipse默认新建JSP的HTML页面,极具参考价值。
    <%@ page language="java" contentType="text/html; charset=ISO-8859-1"
        pageEncoding="ISO-8859-1"%>


    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
    "http://www.w3.org/TR/html4/loose.dtd">
    <html>
    <head>
    <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
    <title>Insert title here</title>
    </head>
    <body>
    </body>
    </html>
    ====================我确实是分割线====================
    其中,只需关注的决定性编码项有:
    ①contentType charset=ISO-8859-1"
    ②pageEncoding="ISO-8859-1"
    ③meta charset=ISO-8859-1"
    所以总的来看,关键就是两个charset与一个pageEnconding
    ====================我确实是分割线====================
    问题的原理如下:
    ①JSP要经过两次的“编码”,

       第一阶段会用pageEncoding,

      第二阶段会用utf-8至utf-8,

      第三阶段就是由Tomcat出来的网页, 用的是contentType。


    ②第一阶段是jsp编译成.java,它会根据pageEncoding的设定读取jsp,结果是由指定的编码方案“翻译”成统一的UTF-8 JAVA源码(即.java),如果pageEncoding设定错了,或没有设定,出来的就是中文乱码。

    ③第二阶段是由JAVAC的JAVA源码至java byteCode的编译,不论JSP编写时候用的是什么编码方案,经过这个阶段的结果全部是UTF-8的encoding的java源码。


    JAVAC用UTF-8的encoding读取java源码,编译成UTF-8 encoding的二进制码(即.class),这是JVM对常数字串在二进制码(java encoding)内表达的规范。


    ④第三阶段是Tomcat(或其的application container)载入和执行阶段二的来的JAVA二进制码,输出的结果,也就是在客户端见到的,这时隐藏在阶段一和阶段二的参数contentType就发挥了功效。


    ====================我确实是分割线====================
    中文乱码解决方案:
    1 调用JSP页面出现乱码 
       通过浏览器调用JSP页面,在浏览器中所有的中文内容出现乱码。 
       首先确认本JSP在编码器中保存人家内容时,使用的UTF-8的编码格式,然后在JSP页面的开咋始,添加<%@ pageEncoding "utf-8"%> 就可以解决这个中文乱码问题。 


    2 调用Servlet页面出现乱码 
       通过浏览器调用Servlet,Servlet在浏览器中所有的中文内容出现乱码 
       在Servlet使用response对输出内容之前,先执行response.setContentType("text/html;charset=utf-8")设定输出的内容的编码。 

    3 POST表单传递参数乱码 
       通过JSP页面,HTML,页面或者Servlet中的表单元素提交参数给对应的JSP页面或者Servlet,而JSP页面或者Servlet接收的中文参数乱码。 
       在接收POST提交的参数之前,使用request.setCharacterEncoding("utf-8")设定接收参数内容的编码; 
       更好的解决方法,就是使用过滤器技术。有很多这样的帖子。 


    4 URI方式传递参数或者GET方式乱码 
       修改TAMCAT的server.xml配置文件, 
       <Connector port="8080" protocol="HTTP/1.1" 
       maxThreads="150" connectionTimeout="20000" 
       redirectPort="8443" URIEncoding="UTF-8" > 
       红色部分就是添加的URIEncoding="UTF-8",重启TAMCAT就行了 

    5 Properties文件乱码 
       在使用一些类库或者框架时(例如:JSTL ,Format标签库,Displaytag标签库和Struts框架),为了实现页面内容的国际化,需要编写对应的properties文件,而propertise文件中的中文内容在显示的时候也会出现乱码。 
       native2ascii -cencoding utf-8 displaytag.properties 
       displaytag_zh_CN.properties 
       还可以使用Eclispse的插件-------Properties Editor
    还有就是数据库的设置。

    JSP(Java Server Page)
        JSP是服务器端运行的页面,JSP本就是一个文档,他不仅可以包含静态的HTML代码,也可以包含动态的JAVA代码,服务器容器可以将JSP转换成Servlet发布,并接受请求。
     Jsp文件只需放在应用文件下就可以了。
     JSP例子:
     <html>
     <head><title>time</title></head>
     <body>
         <h1><%=new java.util.Date()%></h1>
     </body>
     </html>


     JSP运行过程
     1,JSP文件先要翻译成Java文件(Servlet),在tomcat中翻译后的java文件在tomcat下的work\Catalina\localhost中相应名字的应用目录里。
     2,编译Java文件
     3,运行.class文件

     Tomcat的全局配置文件
     在Tomcat下的conf目录中的web.xml是服务器全局的配置文件,其中要想关闭列表显示叫要把
     <init-param>
           <param-name>listings</param-name>
           <param-value>false</param-value>
     </init-param>

     Tomcat的管理员设置
     在tomcat下的conf目录中的tomcat-use.xml文件中可以配置tomcat的用户角色
     <tomcat-users>
       <role rolename="manager" description=""/>
       <role rolename="admin"/><!--设定角色的名字,admin是管理员,manager是项目管理-->
       <user username="manager" password="123" roles="manager"/><!--设定用户名密码,和角色-->
       <user username="admin" password="123" roles="admin"/>
     </tomcat-users>

     JSP的特点
     1,JSP不需要编译
     2,JSP不需要进行配置
     3,JSP本质上就是一个Servlet
     4,JSP是以静态页面为主的。
     
     JSP的语法格式
     <%=...%>表达式,写表达式时不要加上分号,表达式,翻译成Java代码也就是直接使用流输出,也就不能加分号了。
     <%! int a=0; %>声明,使用声明的变量,所创建的变量是在service方法体外,也就是翻译成实例变量,声明也可以声明内部类或者是方法的声明。
     <% if(a==0){ a=3;}%>代码片断,代码段会原样翻译。在代码段中声明的变量,就会被翻译的service的方法体中,也就是被翻译成局部变量。
     前三种JSP的语法也叫做JSP代码,以上的表示方法,都有与之相对应的xml表达形式,因为可读性较差,所以较少使用。
     <%@ ... %>JSP指令
     <%@page import="...,..."%>引入包
     <%include file="...."%>直接引入内容,这些内容会直接嵌入到页面中。

     JSP的注释
     <%--........--%>


     JSP指令和JSP动作(标准的JSP标签)
     JSP指令的语法
     <%@... ....%>
     <%@include file="...."%> 包含指令,包含指令会将要包含的文件代码原样输出到本页面(在翻译时把包含文件的内容引入并合并),不进行翻译,file属性制定要包含的文件的名字(注意要包含的文件的路径),这个指令可以实现页面的复用。
     注意:<%@include file=" "%>要包含的JSP文件中不能定义重名的变量,如果不使用包含指令,那么JSP中的最小转换单元就是JSP文件,使用了包含指令就会将多个jsp的结果合并到一起。
     <%@page ..%>页面指令
     <%@page import="..."%> 引入包 例:<%@page import="java.util.*,ent.*"%>(可出现多次)
     <%@page pageEncoding="GBK"%>设定页面显示的字符集
     <%@page contentType="text/html;charset=GBK"%>设置输出流的编码方式和文件格式。
     注意:

           输出的字符集编码方式要和页面显示的字符集相同,一般定义输出流的编码方式,注意以上两个页面指令在JSP中最好只出现一次。注意在请求中传送的参数的值是中文的话需要调用请求request的setCharacterEncoding("..")方法设置相应的编码方式,只适用于post请求,也可以通过Servlet的Filter来进行中文编码方式的处理。


     <%@page language="java"%>指定JSP的脚本语言,现在只能写java
     <%@page extends=".."> 指定翻译后代码类型的父类,最好不要使用这个指令
     <%@page session="true|false"%>指定是否自动创建Session,最好不自己指定,使用默认值
     <%@page buffer=".." autoflush="true|false"%>设置输出的缓存和是否自动刷新缓存,一般情况下都使用默认值,缓存的默认值为8KB。
     <%@page isThreadSafe="true|false"%>是否线程安全,默认为true,也就是线程安全,只使用默认值。
     <%@page info=".."%>设置页面信息,一般不用
     <%@page errorPage=""%>指定异常处理页面
     <%@page isErrorPage="true|false"%>设定本页面就是异常后跳转的页面
     <%@page isELIgnored="true|false"%>设定本页面是否忽略JSP2.0种EL表达式
     <%@taglib ..%> 标签库指令,用于引入自定义标签库

     JSP动作
     <jsp:... ...>
     <jsp:include page="..." flush=true|false>

              include动作是在翻译和编译后运行时进行响应的合并,也就是对应了Servlet中RequestDispatcher类的include(ServletRequest request, ServletResponse response)方法。
     include动作可以传递参数
     <jsp:include page="test.jsp">
            <jsp:param name="user" value=" ">
     </jsp:include>
     在使用参数时,可以用request内置对象来获取参数的值。
     

    <jsp:forward page="xxx/xxx.jsp">

              forward动作对应的是Servlet中RequestDispatcher类的forward(ServletRequest request, ServletResponse response)方法,也就是把请求进行转发。
     也可以写成
     <jsp:forward>
            <jsp:attribute name="page">xxx/xxx.jsp</jsp:attribute>
     </jsp:forward>
     这种写法等价于上面的写法。
     在使用Servlet的response的重定向sendRedirect(String location)方法在使用时要包含应用的路径,注意,重定向会放弃这个请求,并向服务器发送一个新的请求,但是响应对象还是一个。重定向是用来避免重复提交的。
     注意:对页面的刷新操作就会再一次运行程序,也就仿佛进行了再一次的提交一样,这也就是重复提交的问题,所以需要使用重定向来解决这个问题。
     

    <jsp:plugin  type="bean|applet" code="...">
            ...
     </jsp:plugin>
     jsp:plugin动作可以使页面上运行applet,或者是使用java代码。


     <jsp:useBean id="对象名" class="类全名"|type="类全名"(使用多态时使用)beanName="类全名" scope="page|request|session|application">
          jsp:useBean动作,可以通过这个动作来使用java代码,id就是变量名,class和type的区别在于,使用type是不会创建对象,而只是会先查找如果有该对象就会使用,如果没有就会抛出异常,class在没有该对象存在时,则会创建新对象。scope是用来保存实例的空间范围,page、request、session、application,由小到大,默认为page。

     <jsp:setProperty name="对象名" property="属性名" value="属性值"/>
     直接指定Bean属性的值,必须类型一致。
     <jsp:setProperty name="对象名" property="属性名" param="from表单中的响应的参数名"/>
     用表单中指定的参数来为Bean属性赋值,系统会自动作类型转换。
     <jsp:setProperty name="对象名" property="属性名"/>
     用表单中和Bean属性同名的标单项来给Bean属性赋值,这样写只会为指定的属性赋值
     <jsp:setProperty name="对象名" property="*"/>
     这种方式会自动用表单中的同名参数来为Bean的所有属性赋值
     <jsp:setProperty name="对象名" property="属性名" value="<%= ...%>"/>
     使用表达式来为Bean的属性进行赋值


     <jsp:getProperty name="对象名" property="属性名">
            取得指定的属性,这个动作不常用,一般用表达式代替
     我们可以通过设置属性名的方法名也就是setXxxx()来控制属性的赋值,用form表单的参数为Bean的属性赋值时,系统会自动的调用与之同名属性的setXxxx()方法,尤其是日期类型,可以通过使用这一方法来处理。
     例:
     import java.sql.*;
     public class Person{
            private Srting name;
            private Date birthday;
            public void setName(String name){
                   this.name=name;
            }
            public void setBirthdayAsString(String birthday){
                   this.setBirthday(valueOf(birthday));
            }
            public void setBirthday(Date birthdays){
                   this.birthday=birthday;
            }
     }
     add.jsp
     <jsp:useBean id="person" type="Person" scope="request">
     <jsp:setProper name="person" property="*"/>
     </jsp:useBean>
     toadd.jsp
     <form action="xxx" methmod="post">
            <input type="text" name="name">
            <input type="text" name="birhtdayAsString">
            <input type="submit" value="add.jsp">  
     </form>


     JavaBean和EJB的区别
     JavaBean是可视的或者是不可视的,不用部署,EJB是不可视的,需要部署到服务器中。
     JavaBean部署在应用程序中,EJB是部署在容器中,可以是有状态或者是无状态的,声明式事务。
     JavaBean的属性和方法是普通的,EJB是需要遵守规范,是需要通过标准的描述符来定义的。
     JavaBean是基于事件驱动的,EJB中只有消息驱动Bean是基于事件的
        注意:jsp:useBean动作,用表单为Bean的属性赋值时,也就是jsp:setProperty动作直接收请求中表单的信息为使用的Bean的属性进行赋值,也就是说jsp:useBean和jsp:setProperty不能够出现在为这个Bean的属性赋值的form表单的页面上。
     

    JSP的异常处理
     <%@page errorPage="xxx.jsp"%> 指定本页面出现异常后要转到的页面
     <%@page iserrorPage="true"%> 见本页面指定为异常处理页面,也就是其他出异常的页面可以指定本页面为要转到的异常处理页面。

     处理异常
      运行时异常
      1) 一个程序运行时重获的错误;
      2) 能被一固有的异常对象引用所使用;
      3) 可以利用异常引用创建或产生一个JSP页面显示异常信息给用户。
     

    创建一个异常跟踪方案
     1) 在每一个JSP页面,包含要创建的异常页面名称;
     <%@page import="numberguess.NumberGuessBean" errorPage="error.jsp"%>
     2) 创建一个异常页面;
     <%@page isErrorPage="true"%>
     JSP页面中的异常处理
     1) <%=exception.toString()%>        打印异常名称;
     2) <%exception.printStackTrace();%> 打印当前错误流中所有错误的列表;
     3) <%=exception.getMessage()%> 打印错误的详细描述信息

     JSP中异常页面的部署描述
     <error-page>
            <error-code>404</error-code>
            <location>notFound.html</location>
     </error-page>
     <error-page>
            <exception-type>java.lang.NumberFormatException</exception-page>
     <location>error.jsp</location>
     </error-page>

     JSP隐含内置对象
     JSP中隐含内置对象
     名称         类型                                     注释和范围
     request      javax.servlet.http.HttpServletRequest                    request
     response     javax.servlet.http.HttpServletResponse                  response
     page         javax.lang.Object                                 page
     Exception    java.lang.Throwable                               page
     pageContext  javax.servlet.jsp.PageContext                          page
     session      javax.servlet.http.HttpSession                         session
     application  javax.servlet.ServletContext                     ServletContext
     out          javax.servlet.jsp.JspWriter                          OutputStream
     config       javax.servlet.ServletConfig                           ServletConfig

     内置对象request(HttpServletRequest)
     1)封装了来自客户端的请求: 
     2)方法:
     Parameter related methods:
          +getParameterNames():Enumberation 返回form中对象名称的枚举;       
           +getParameter(String name):String返回指定名称的对象值;
           +getParameterValues(String name) :String[ ] 返回指定名称的对象值数组;
            +getParameterMap():Map
     Attribute related methods:
            +setAttribute(String name,Object value) :void  设置属性
            +getAttribute(String name) :Oject  返回属性值;
            +removeAttribute(String name) :void
            +getAttributeNames():Enumberation  返回属性名称
     URL related methods:
            +getContextPath():String
            +getServletPath():String
            +getPathInfo():String
     State related methods:
            +getCookies():Cooie[ ] 取得Cookie数组;
            +getSession():HttpSession
            +getSession(Boolean create) :HttpSession
     
     内置对象response(HttpServletResponse)
             +getWriter():PrintWriter
             +addCookie(Cookie c) :void
             +encodeURL(String URL) :String
             +sendRedirect(String URL) :void
             +sendError(int errorCode) :void
     
     内置对象session(HttpSession)
            +setAttribute(String name,Object value) :void  设置属性
            +getAttribute(String name) :Oject  返回属性值;
            +removeAttribute(String name) :void
            +getAttributeNames():Enumberation  返回属性名称
     Other motheds:
            +seMaxInactiveInterval(int seconds) :void
             Invalidate():void
     
     内置对象pageContext(PageContext)
            +setAttribute(String name,Object value) :void  设置属性
            +getAttribute(String name) :Oject  返回属性值;
            +removeAttribute(String name) :void
            +getAttributeNames():Enumberation  返回属性名称
            +findAttribute(String name) :Object
     Other motheds:
     +getRequest():HttpServletRequest
     +getSession():HttpSession
     +getServletContext():ServletContext
     +getOut():JspWriter
     
     内置对象application(ServletContext)
            +setAttribute(String name,Object value) :void  设置属性
            +getAttribute(String name) :Oject  返回属性值;
            +removeAttribute(String name) :void
            +getAttributeNames():Enumberation  返回属性名称
     Other methods:
           +getRequstDispather(String URL):RequestDispather
         +getInitParameter(String name) :String


     内置对象out
     1)JSPWriter的一个实例, 用于发送响应给客户端 
     2)方法:
        print(String)/println(String) 用于向页面输出       
        print(int)/println(int)       
        flush() 刷新out对象的缓存


     内置对象exception
     1) 用于异常对象的异常实例; 
     2) 异常页面在page指令中存在属性isErrorPage="true"; 
     3) 方法:
        getMessage()/printStackTrace()/toString()

     内置对象session
     1) HttpSession的一个实例, 用于来维护session的状态 
     2) 方法:
        getAttribute(String name)/setAttribute(String name)
        取得设置的属性的对应资源/设置属性的对应资源          
        removeAttribute(String name) 删除属性和队应资源          
        getAttributeNames()
        返回值是一个迭代器,可以获得session中设置的属性的名字也就是在set时使用的与资源对应的那个名字

     内置对象application
     1) ServletContext一个实例,用来维护application的状态 
     2) 方法:
        getAttribute()/setAttribute() 和session以及request中的那一对设置属性及其对应资源的方法 
        getInitParameter()   获得初始化参数             
        getServletInfo()  获得Servlet的信息
     3) application的作用域比session大得多,一个session和一个client联系, 而application保持了用于所有客户端的状态。
          注意:如果要保存的数据量比较大时,就要使用生命周期比较短的request对象,page对象只在本页面中有效,也就是会再翻译后的service方法体内的一个局部变量,service运行结束后就会销毁,不像request可以转发。session和application的生命周期都比较长而且作用域比较大。
     

    JSP自定义标签(tag)
     什么是自定义标签?
     1,用户自定义的Java语言元素, 实质是运行一个或者两个接口的JavaBean;
     2,可以非常紧密地和JSP的表示逻辑联系在一起,又具有和普通JavaBean相同的业务逻辑处理能力;
     3,当一个JSP页面转变为servlet时,其间的用户自定义标签转化为操作一个称为标签hander的对象;
     4,可操作默认对象,处理表单数据,访问数据库以及其它企业服务;

     自定义标签库的特点
     1,通过调用页面传递参数实现定制;
     2,访问所有对JSP页面可能的对象;
     3,修改调用页面生成的响应;
     4,自定义标签间可相互通信;
     5,在同一个JSP页面中通过标签嵌套,可实现复杂交互。

     如何使用自定义标签库
     1,声明标签库
     2,使标签库执行对Web应用程序可用

     声明标签库
     1,使用taglib指令声明标签库
     2,语法:%@taglib uri="URI" prefix="pre" %

     注意:a. uri属性可以是绝对的,也可以是相对URL,该URL指向标记库描述符(TLD)文件;   
           b. uri属性也可以是一个并不存在的URL,该URL为web.xml文件中将标记库描述符(TLD)文件的绝对
     URL到本地系统的一个映射;
     3,范例:<%@taglib uri="/WEB-INF/template.tld" prefix="test" %>            
              <%@taglib uri="http://java.sun.com/jstl/core" prefix="core" %>    

     使标签库执行可用
     方式一:在WEB-INF/classes目录下部署标记处理程序类;
     方式二:将标记处理程序类打包成jar文件并置于WEB-INF/lib目录。

     几种典型的标签
     1,不带属性和主体的简单标签:<mytaglibs:SomeTag/>;
     2,不带主体但有属性的标签:<mytaglibs:SomeTag user="TonyDeng"/>;
     3,带有主体和属性的标签:
              <mytaglibs:SomeTag user="TonyDeng">                                                              
                       ...// 标签体                                
              </mytaglibs:SomeTag>;
     注意:a. 属性列于start tag中,它是在标记库描述符(TLD)文件中指定,服务于标记库的自定义行为;     
            b. 标签体位于start tag和end tag间,可以是任何合法的JSP内容或者标签;
     

    定义标签
     1,开发实现tag的类(tag handler);
     2,编辑标记库描述符(TLD)文件;
     3,在web.xml中为标记库描述符(TLD)文件的绝对URL建立一个映射(该步骤可选);

     标记库描述符(TLD)文件
     1,一个描述标记库的XML文件;
     2,内容开始是整个库的描述,然后是tag的描述;
     3,标记库描述符(TLD)文件用于Web Container确认tag以及JSP页面发展工具;

     实现tag的类(标签处理器)(tag handler)
     1,是一些在引用了标签的JSP页面执行期间被Web Container调用以求自定义标签值的对象;
     2,必须实现Tag, SimpleTag和BodyTag之一;
     3,可以继承TagSupport和BodyTagSupport之一。
     标签库的接口和类的继承关系接口的继承关系:    
     ☉interface javax.servlet.jsp.tagext.JspTag        
             ☉interface javax.servlet.jsp.tagext.SimpleTag          
             ☉interface javax.servlet.jsp.tagext.Tag          
                    ☉interface javax.servlet.jsp.tagext.IterationTag          
                    ☉interface javax.servlet.jsp.tagext.BodyTag
     类的继承关系:    
              ☉class javax.servlet.jsp.tagext.TagSupport
               (implements javax.servlet.jsp.tagext.IterationTag, java.io.Serializable)        
                     ☉class javax.servlet.jsp.tagext.BodyTagSupport
                      (implements javax.servlet.jsp.tagext.BodyTag)     
              ☉class javax.servlet.jsp.tagext.SimpleTagSupport
                 (implements javax.servlet.jsp.tagext.SimpleTag)
                       (Interface)                 
                          JspTag                   
                             |                           
                     |ˉˉˉˉˉˉˉˉ|           
                (Interface)      (Interface)JSP2.0          
                    Tag           SimpleTag ←--SimpleTagSupport            
                     |                                        
                     |          
                 (Interface)         
                IterationTag←--TagSupport
                支持迭代的标签         |
                     |                 |        
                     |                 | 
                 (Interface)           |             
                  BodyTag ←---BodyTagSupport
                 可以处理标签体

     自定义标签的开发步骤
     1,写标签处理器,也就是一个符合自定义标签规范的类xxx.java
     2,写标签库定义文件,也就是定义标签的格式规范,也要符合自定义标签的规范xxx.tld
     3,建上述两步中的相应文件进行部署。web.xml
     tld文件中的配置
       <tag>
         <name>loop</name>
         <tag-class>com.tag.LoopTag</tag-class>
         <body-content>jsp</body-content>
         <!--设定自定义标签体内的内容,可以置为empty,也就是只能写空标签-->
         <attribute>
           <name>counter</name><!--配置自定义标签的属性名-->
           <required>true</required><!--配置属性是否必须出现-->
           <rtexprvalue>true</rtexprvalue><!--允许使用表达式作为属性的值-->
           <type>int<type><!--配置属性的类型-->
           </attribute>
       </tag>

     简单的标签处理程序类
     1,必须实现Tag接口的doStartTag()和doEndTag()方法;
     2,因为不存在Body,doStartTag()方法必须返回SKIP_BODY;
     3,如其余页面要执行,doEndTag()方法返回EVAL_PAGE, 否则返回SKIP_PAGE;
     4,对于每一个标签属性,你必须在标签处理程序类里定义一个特性以及get和set方法以一致于JavaBeans   体系惯例
     

    带Body的自定义标签
     1,必须实现Tag接口的doStartTag()和doEndTag()方法;
     2,可以实现IterationTag接口的doAfterBody()方法;
     3,可以实现BodyTag接口的doInitBody和setBodyContent方法;
     4,doStartTag方法可以返回SKIP_BODY、EVAL_BODY_INCLUDE、或者EVAL_BODY_BUFFERED(当你想使用     BodyContent);
     5,doEndTag方法可以返回SKIP_PAGE或EVAL_PAGE;
     6,doAfterBody方法可以返回EVAL_BODY_AGAIN, SKIP_BODY;


     定义脚本变量的标签(迭代标签)
     1,定义脚本标签的二个步骤:
        . 在标记库描述符(TLD)文件中列明脚本变量;
        . 定义标签扩展信息类(TEI)并且在TLD文件中包括这个类元素(tei-class);
     2,变量必须在标签处理程序类中使用pageContext.setAttribute()方法设置;
     3,标签扩展信息类(TEI)必须继承TagExtraInfo以及覆盖getVariableInfo()方法;
     4,变量的范围可以是AT_BEGIN, NESTED, AT_END(标签扩展信息类(TEI)的VariableInfo中定义)之一;
     脚本变量的有效性
     变量     |   有效性                 
     ---------------------------------------
     NESTED   | 标签中的参数在starttag到endtag之间是有效的
     AT_BEGIN | 标签中的参数在标签的开始到JSP页面结束是有效的
     AT_END   | 标签中的参数在标签的结束到JSP页面结束是有效的

     Tag接口的方法
     interface javax.servlet.jsp.tagext.Tag
     ------------------------------------------------------
     +EVAL_BODY_INCLUDE:int
     +EVAL_PAGE:int//继续执行页面
     +SKIP_BODY:int//跳出标签体
     +SKIP_PAGE:int//跳出页面,也就是整个JSP不会再运行
     ------------------------------------------------------
     +release():void
     +getParent():javax.servlet.jsp.tagext.Tag
     +setParent(javax.servlet.jsp.tagext.Tag):void
     +doEndTag():int
     +doStartTag():int
     +setPageContext(javax.servlet.jsp.PageContext):void

     Tag的生命周期
     1,setPageContext(javax.servlet.jsp.PageContext):void
     2,setParent(javax.servlet.jsp.tagext.Tag):void
     3,setAttribute:void
     4,doStartTag():int
     5,doEndTag():int
     6,release():void

     BodyTag和Tag接口的关系
     interface javax.servlet.jsp.tagext.BodyTag-->interface javax.servlet.jsp.tagext.Tag
     ------------------------------------------
     +EVAL_BODY_AGAIN:int//继续执行标签体
     -----------------------------------------
     +doInitBody():void
     +setBodyContent(javax.servlet.jsp.tagext.BodyContext):void
     +doAfterBody():int


     BodyTag的处理过程
     1,setPageContext(javax.servlet.jsp.PageContext):void
     2,setParent(javax.servlet.jsp.tagext.Tag):void //设置此标签的父标签,也就是上一层标签
     3,setParent()
     4,doStartTag():int
     5,setBodyContent(javax.servlet.jsp.tagext.BodyContent):void//设置方法体
     6,doInitBody():void//
     7,doAfterBody():int
     /* 这个方法会被不断的调用直到不再满足条件,通过这个方法来控制返回,从而控制循环是否中断
       SKIP_BODY是跳出标签体的,EVAL_BODY_INCLUDE是继续执行标签体,通过返回这两个值来控制循环是否*/
     8,doEndTag():int
     9,release():void

     SimpleTag接口的方法
     javax.servlet.jsp.tagext.SimpleTag
     ------------------------------------------
     +doTag():void
     +getParent():JspTag
     +setJspBody(JspFragment jspBody):void
     +setJspContext(JspContext pc):void
     +setParent(JspTag parent):void
     SimpleTage接口的生命周期(JSP2.0的自定义标签接口)

     1,new:    
        每次遇到标签,容器构造一个SimpleTag的实例,这个构造方法没有参数。和红典的标签一样,          SimpleTag不能缓冲,故不能重用,每次都需要构造新的实例。
     2,setJspContext()、setParent(): 只有这个标签在另一个标签之,才调用setParent()方法;
     3,设置属性:调用每个属性的setter方法;
     4,setJspBody();
     5,doTag(): 所有标签的逻辑、迭代和Body计算,都在这个方法中;
     6,return


     例:
     package com.tag;
     public class HelloTag implements Tag
     {
             private PageContext pageContext;
             private Tag parent;
          public HelloTag(){
               super();
          }
       
         /**
           *设置标签的页面的上下文
           */
          public void setPageContext(final PageContext pageContext) {
                this.pageContext=pageContext;
          }
         /**
           *设置上一级标签 www.2cto.com
           */
          public void setParent(final Tag parent) {
               this.parent=parent;
          }
          /**
           *开始标签时的操作
           */
          public int doStartTag() throws JspTagException{
               return SKIP_BODY;  //返回SKIP_BODY,表示不计算标签体
          }
          /**
           *结束标签时的操作
           */
          public int doEndTag() throws JspTagException{
                 try{
                      pageContext.getOut().write("Hello World!你好,世界!");
                 }                
               catch(java.io.IOException e){
                    throw new JspTagException("IO Error: " + e.getMessage());
               }
               return EVAL_PAGE;
           }
          /**
           *release用于释放标签程序占用的资源,比如使用了数据库,那么应该关闭这个连接。
           */
          public void release() {} 
          public Tag getParent(){ 
             return parent;
          }
     }
     xxx.tld
     <?xml version="1.0" encoding="ISO-8859-1" ?>
     <taglib xmlns="http://java.sun.com/xml/ns/j2ee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee web-jsptaglibrary_2_0.xsd"
         version="2.0">
       <tag>
         <description>Extends TagSupport</description>
         <name>hello</name>
         <tag-class>com.tag.HelloWorldTag</tag-class>
         <body-content>jsp</body-content>
       </tag>
     </taglib>


     web.xml
     <web-app>
     <taglib-uri>/xxx</taglib-uri>
            <taglib-location>/WEB-INF/tlds/xxx.tld</taglib-location>
     </web-app>


     xxx.jsp
     <%@ taglib uri="/xxx" prefix="mytag" %>
     <%@ page contentType="text/html; charset=gb2312" %>
     <html><head><title>first cumstomed tag</title></head><body>
     <p>以下的内容从Taglib中显示:</p>
     <p><i><mytag:hello_int/></i>
     <br>
     <p><mytag:hello_int></mytag:hello_int>
     </body></html>


     JSP2.0中的表达式语言(EL表达式)
     EL语法
     EL的语法很简单,他最大的特点就是使用上很方便
     例:
     ${sessionScope.user.sex}
     所有EL都是以${ 为起始、以} 为结尾的。
     上述EL范例的意思是:从Session取得用户的性别。如果使用之前JSP代码的写法如下:
     <%
       User user = (User)session.getAttribute("user");
       String sex = user.getSex( );
      %>
     两者相比较之下,可以发现EL的语法比传统JSP代码更为方便、简洁。
     EL提供. 和[ ] 两种运算符来存取数据,[ ]可以访问集合或者是数组的元素、Bean的属性。下列两者所代表的意思是一样的,但是需要保证要取得对象的那个的属性有相应的setXxx()和getXxx()方法才行。
     例:
     ${sessionScope.user.sex}
     等于
     ${sessionScope.user["sex"]}
     . 和[ ] 也可以同时混合使用,如下:
     ${sessionScope.shoppingCart[0].price}
     返回结果为shoppingCart中第一项物品的价格。
     在EL中,字符串既可以使用"abc",可以使用'abc'。


     EL运算符
     EL的算术运算符和Java中的运算符的大致相同,优先级也相同。
     注意:'+' 运算符不会连接字符串了,他只用于加法运算。
     EL关系运算符有以下六个运算符
     关系运算符      说 明                范 例             结果
     = = 或eq   |    等于   |${ 5 = = 5 } 或${ 5 eq 5 } | true
     != 或ne    |   不等于  |${ 5 != 5 } 或${ 5 ne 5 }  | false
     < 或lt     |   小于    |${ 3 < 5 }或${ 3 lt 5 }    | true
     > 或gt     |   大于    |${ 3 > 5 }或${ 3 gt 5 }    | false
     <= 或le    |  小于等于|${ 3 <= 5 }或${ 3 le 5 }   | true
     >= 或ge    |  大于等于|${ 3 >= 5 }或${ 3 ge 5 }   | false


     empty运算符
     Empty运算符主要用来判断值是否为null或空的,例如:
     ${ empty param.name }
     接下来说明Empty运算符的规则:
     {empty} A
        如果A为null时,返回true
        如果A不存在时,返回true
        如果A为空字符串时,返回true
        如果A为空数组时,返回true
        如果A为空的Map时,返回true
        如果A为空的Collection时,返回true
        否则,返回false
     注意:
     在使用EL关系运算符时,不能够写成:
     ${param.password1} = = ${param.password2}
     或者
     ${ ${param.password1 } = = ${ param.password2 } }
     而应写成
     ${ param.password1 = = param.password2 }
     使用EL从表单中取得数据
     与输入有关的隐含对象有两个:param和paramValues,它们是EL中比较特别的隐含对象。一般而言,我们在取得用户的请求参数时,可以利用下列方法:
     request.getParameter(String name)
     request.getParameterValues(String name)
     在EL中则可以使用param和paramValues两者来取得数据。
     ${param.name}
     ${paramValues.name}可以取得所有同名参数的值
     ${paramValues.hobbies[0]}可以通过指定下标来访问特定的参数的值
     这里param的功能和request.getParameter(String name)相同,而paramValues和request.getParameterValues(String name)相同。如果用户填了一个form,form名称有为username的文本框,则我们就可以使用${param.username}来取得用户填入文本框的值。


     EL函数
     EL中使用函数要写一个要使用到方法的类,然后在配置xxx.tld文件,然后在JSP中使用时和JSP的自定义标签相似。
     xxx.tld中的配置
     <function>
       <name>reverse</name><!--函数名-->
       <function-class>jsp2.examples.el.Functions</function-class><!--函数所在的类-->
       <function-signature>java.lang.String reverse( java.lang.String )</function-signature>
       <!--函数原型,也就是函数的返回值类型,函数名,参数表,注意一定要写类型的全名-->
     </function>

     使用EL函数的写法
     ${sn:upper('abc')}
     注意:在定义EL函数时,都必须为公开静态(public static)

     EL的隐含对象
     EL也可以使用内置对象中设置的属性,需要使用特定的EL内置对象
     属性范围     |    在EL中的对象
     Page         |     pageScope
     Request      |     requestScope
     Session      |     sessionScope
     Application  |     applicationScope

     EL中使用内置对象的属性
     ${requestScope.user}
     等价于
     <%request.getAttribute("user")%>
     如果不写出特定的范围 ,那就会在不同的范围间进行搜索了
     例:{user}(user是在request范围request.setAttribute("user",user))
     也就等于
     ${requestScope.user}
     <%request.getAttribute("user")%>

     EL的隐含对象
        对象                        类 型                  说 明
     PageContext      |  javax.servlet.ServletContext |表示此JSP的PageContext
     -----------------------------------------------------------------------------------
     PageScope        |         java.util.Map         |取得Page范围的属性名称所对应的值
     -------------------------------------------------------------------------------------
     RequestScope     |         java.util.Map         |取得Request范围的属性名称所对应的值
     -------------------------------------------------------------------------------------
     sessionScope     |         java.util.Map         |取得Session范围的属性名称所对应的值
     -------------------------------------------------------------------------------------
     applicationScope |         java.util.Map         |取得Application范围的属称所对应的值
     -------------------------------------------------------------------------------------
     param           |         java.util.Map         |如同ServletRequest.getParameter(String                      |                               |name)返回String类型的值
     ----------------------------------------------------------------------------------------
     paramValues      |         java.util.Map         |如同ServletRequest.getParameterValues                       |                               |(String name)。返回String []类型的值
     -----------------------------------------------------------------------------------------
     header           |         java.util.Map         |如同ServletRequest.getHeader(String name)            |                               |返回String类型的值
     ------------------------------------------------------------------------------------------
     headerValues     |         java.util.Map         |如同ServletRequest.getHeaders(String name)                  |                               |。返回String []类型的值
     ------------------------------------------------------------------------------------------
     cookie                |     java.util.Map         |如同HttpServletRequest.getCookies( )
     ------------------------------------------------------------------------------------------
     initParam             |  java.util.Map         |如同ServletContext.getInitParameter(String                |                               |name)。返回String类型的值
     -------------------------------------------------------------------------------------------

     cookie对象
              所谓的cookie是一个小小的文本文件,它是以key、value的方式将Session Tracking的内容记录在这个文本文件内,这个文本文件通常存在于浏览器的暂存区内。JSTL并没有提供设定cookie的动作,因为这个动作通常都是后端开发者必须去做的事情,而不是交给前端的开发者。如果我们在cookie中设定一个名称为userCountry的值,那么可以使用${cookie.userCountry}来取得它。


     header和headerValues(请求报头对象)
          header储存用户浏览器和服务端用来沟通的数据,当用户要求服务端的网页时,会送出一个记载要求信息的标头文件,例如:用户浏览器的版本、用户计算机所设定的区域等其他相关数据。如果要取得用户浏览器的版本,即${header["User-Agent"]}。另外在很少机会下,有可能同一标头名称拥有不同的值,此时必须改为使用headerValues来取得这些值。
     注意:因为User-Agent中包含“-”这个特殊字符,所以必须使用“[]”,而不能写成${header.User-Agent}。
     
     initParam
     就像其他属性一样,我们可以自行设定web应用的环境参数(Context),当我们想取得这些参数时,可以使用initParam隐含对象去取得它,例如:当我们在web.xml中设定如下:
     <?xml version="1.0" encoding="ISO-8859-1"?>
     <web-app xmlns="http://java.sun.com/xml/ns/j2ee"
     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
     xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"
     version="2.4">
       <context-param>
          <param-name>userid</param-name>
          <param-value>mike</param-value>
       </context-param>
     </web-app>
     那么我们就可以直接使用${initParam.userid}来取得名称为userid,其值为mike的参数。下面是之前的做法:String userid = (String)application.getInitParameter("userid");
     pageContext对象
     我们可以使用${pageContext}来取得其他有关用户要求或页面的详细信息。下面列出了几个比较常用的部分。
     Expression                                说 明
     ${pageContext.request}                  |取得请求对象
     ${pageContext.session}                  |取得session对象
     ${pageContext.request.queryString}      |取得请求的参数字符串
     ${pageContext.request.requestURL}       |取得请求的URL,但不包括请求之参数字符串
     ${pageContext.request.contextPath}      |服务的web application的名称
     ${pageContext.request.method}           |取得HTTP的方法(GET、POST)
     ${pageContext.request.protocol}         |取得使用的协议(HTTP/1.1、HTTP/1.0)
     ${pageContext.request.remoteUser}       |取得用户名称
     ${pageContext.request.remoteAddr }      |取得用户的IP地址
     ${pageContext.session.new}              |判断session是否为新的,所谓新的session,表示刚由                                                server产生而client尚未使用
     ${pageContext.session.id}               |取得session的ID
     ${pageContext.servletContext.serverInfo}|取得主机端的服务信息
     

    JSTL(JSP标准标签库)

           JSTL由核心标签,<c:... > ,

          xml解析标签<x:...>,

           国际化标签<fmt:....>,

          数据库访问标签<sql:...>,

           函数标签<fn:...>


     核心标签
     Core
     <%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
     属性设置
     <c:set> 设置属性
     <c:remove> 移除设置的属性


     过程控制
     <c:if test="..."> 条件标签 只有在test属性的值为true是才会执行标签体
     例:
     <c:if test="${!(empty user.age)}">
           <h1>hello</h1>
     </c:if>


     <c:choose>choose和when是组合在一起使用的,有点类似于swith case的语法  。
     <c:when test="...">when也是条件判断标签,test属性的值为true是才会执行标签体。
     例:
     <c:choose>
      <c:when test="${param.age<18}">
        <h1>you is a child<h1>
      </c:when>
      <c:when test="${param.age>18 and param.age<50 }">
         <h1>you is a young person</h1>
      </c:when>
      <c:when test="${param.age>50}">
          <h1>you is a old person</h1>
      </c:when>
     </c:choose>


     <c:forEach>迭代标签
     例:
     <c:forEach var="book" item="${store.books}" varStatus="status">
         <h1>${book.parice}</h1>
     </c:forEach>
     <c:forEach begin="1" end="5" step="1">
          <h1>hello</h1>
     </c:forEach>

     <c:forTokens>字符串迭代标签


     <c:import>引入标签
     <c:import url="引入内容的url" var="别名">
     ${别名}
     <c:url>url标签
     <c:url value="...">
      <c:param name="..." value="..."/>
     </c:url>
     <c:url value="...">
     使用url标签可以实现URL回写
     <c:redirect uri="xxx/xxx/xxx.xx"/>


     国际化标签
     <fmt:lauguage>
     <fmt:bundel>资源指定标签
     <fmt:message>消息标签
     例:
     <fmt:setLocale value="zh"/>
     <fmt:bundel basename="message.MessageResources">
       <fmt:message>name</fmt:message>
     </fmt:bundel>
     xxxx.properties
     name=\0060\0700\
     

       在核心JSP标记上使用JSTL有以下几大优势。

       首先,JSTL现在采用独立于JSP标记的规范来定义。因此,即使JSP规范发生改动,也可以保证JSTL标记的一致性。容易受到JSP容器变更影响的许多应用程序都将受益于JSTL规范.

       其次,JSTL具有比核心JSP标记更强大的功能,正如您在最佳实践系列中所视, c:import不仅 可以导入其它站点的URL,而且,它还可以导入 jsp:include 标记不可用的更为复杂的内容。

       最后,JSTL使用自己的表达式语言(通常简称为EL)。JSTL EL可以相当灵活地写入JSP代码,是对核心JSP语言的有力补充。

      

    JSTL的 c:import 在模仿JSP的include原理。但是,如果深入研究 c:import (或任何JSTL标记),您将发现许多其它功能。除了分配参数并管理本地文件内容以外, c:import 还可以引入外部站点的内容。

       类似于 jsp:includec:import 使用各种参数以便可在Web站点灵活移动内容。但与include指令不同, c:import 并不限制访问本地文件。

    jsp:include:

    <jsp:include page="header.jsp" flush="true">
         <jsp:param name="pageTitle" value="newInstance.com"/>
         <jsp:param name="pageSlogan" value=" " />
    </jsp:include>

    JSTL C:import

    <c:import url="header.jsp">
         <c:param name="pageTitle" value="newInstance.com"/>
         <c:param name="pageSlogan" value=" " />
    </c:import>

    c:paramjsp:param 非常相似。

         使用 c:import 的真正优势在于,它可以引入外部Web站点的内容或Web应用程序。在前面学习jsp:includes时,您可能已经注意到,我们使用file(文件)属性来为include指定静态内容。file(文件)属性正如其名称所示:使您可以引入本地文件的内容。 c:import 的相应属性为 url ,它也正如其名称所示:使您能够引入任何URL。 c:import 不仅可以使用本地文件的内容来填充您的站点网页,而且还允许您引入任何URL的内容,使其它站点的内容可以真正灵活地适应您自己站点的外观和风格。

    <%@ page language="java" contentType="text/html" %>
    <%@ taglib prefix="c" uri="http://java.sun.com/jstl/core" %>
    <html>
    <head>
         <title>newInstance.com</title>
         <meta http-equiv="Content-Type" 
           content="text/html; charset=iso-8859-1" />
         <link href="/styles/default.css" rel="stylesheet" type="text/css" />
    </head>
    <body>
    <c:import url="header.jsp">
         <c:param name="pageTitle" 
           value="newInstance.com :: True North Guitars"/>
         <c:param name="pageSlogan" value="...building it from scratch" />
    </c:import>
    <%@ include file="/navigation.jsp" %>
    <c:import url="bookshelf.jsp" />
    <c:import 
      url="http://www.truenorthguitars.com/Clients/Richman/index.htm" />
    <%@ include file="/footer.jsp" %>
    </body>
    </html>
    

           这些代码看上去没有问题——但是,如果您试着在自己站点运行,您很快就会发现问题。所有图像都不显示,并且相关链接全部出现故障。当然,如果仔细想想,原因很很简单。因为外部资源(本例中是图像文件)被解释(interpreted),而您将解释结果直接导入您的输出流。外部图像文件链接如 /images/guitar-01-24.jpg 将显示不存在。解决该问题的唯一方法是将原图像复制到您的站点,与简单的导入相比,这是一项完全不同(并且消耗更多时间)的技术。

    因此,最好知道要导入的外部内容是否是纯文本。例如,要考虑连接到系统管理员的站点及其本身的头文件、脚注文件、外观和风格。该站点的网页详细说明如何设置 README 文件,以便用户可以查看如何使用该站点。

    JSP的内置对象及方法

    JSP的内置对象及方法


    1.request对象
    客户端的请求信息被封装在request对象中,通过它才能了解到客户的需求,然后做出响应。它是HttpServletRequest类的实例。
    1 object getAttribute(String name) 返回指定属性的属性值
    2 Enumeration getAttributeNames() 返回所有可用属性名的枚举
    3 String getCharacterEncoding() 返回字符编码方式
    4 int getContentLength() 返回请求体的长度(以字节数)
    5 String getContentType() 得到请求体的MIME类型
    6 ServletInputStream getInputStream() 得到请求体中一行的二进制流
    7 String getParameter(String name) 返回name指定参数的参数值
    8 Enumeration getParameterNames() 返回可用参数名的枚举
    9 String[] getParameterValues(String name) 返回包含参数name的所有值的数组
    10 String getProtocol() 返回请求用的协议类型及版本号
    11 String getScheme() 返回请求用的计划名,如:http.https及ftp等
    12 String getServerName() 返回接受请求的服务器主机名
    13 int getServerPort() 返回服务器接受此请求所用的端口号
    14 BufferedReader getReader() 返回解码过了的请求体
    15 String getRemoteAddr() 返回发送此请求的客户端IP地址
    16 String getRemoteHost() 返回发送此请求的客户端主机名
    17 void setAttribute(String key,Object obj) 设置属性的属性值
    18 String getRealPath(String path) 返回一虚拟路径的真实路径

    2.response对象
    response对象包含了响应客户请求的有关信息,但在JSP中很少直接用到它。它是HttpServletResponse类的实例。
    1 String getCharacterEncoding() 返回响应用的是何种字符编码
    2 ServletOutputStream getOutputStream() 返回响应的一个二进制输出流
    3 PrintWriter getWriter() 返回可以向客户端输出字符的一个对象
    4 void setContentLength(int len) 设置响应头长度
    5 void setContentType(String type) 设置响应的MIME类型
    6 sendRedirect(java.lang.String location) 重新定向客户端的请求

    3.session对象
    session对象指的是客户端与服务器的一次会话,从客户连到服务器的一个WebApplication开始,直到客户端与服务器断开连接为止。它是HttpSession类的实例.
    1 long getCreationTime() 返回SESSION创建时间
    2 public String getId() 返回SESSION创建时JSP引擎为它设的惟一ID号
    3 long getLastAccessedTime() 返回此SESSION里客户端最近一次请求时间
    4 int getMaxInactiveInterval() 返回两次请求间隔多长时间此SESSION被取消(ms)
    5 String[] getValueNames() 返回一个包含此SESSION中所有可用属性的数组
    6 void invalidate() 取消SESSION,使SESSION不可用
    7 boolean isNew() 返回服务器创建的一个SESSION,客户端是否已经加入
    8 void removeValue(String name) 删除SESSION中指定的属性
    9 void setMaxInactiveInterval() 设置两次请求间隔多长时间此SESSION被取消(ms)

    4.out对象
    out对象是JspWriter类的实例,是向客户端输出内容常用的对象
    1 void clear() 清除缓冲区的内容
    2 void clearBuffer() 清除缓冲区的当前内容
    3 void flush() 清空流
    4 int getBufferSize() 返回缓冲区以字节数的大小,如不设缓冲区则为0
    5 int getRemaining() 返回缓冲区还剩余多少可用
    6 boolean isAutoFlush() 返回缓冲区满时,是自动清空还是抛出异常
    7 void close() 关闭输出流

    5.page对象
    page对象就是指向当前JSP页面本身,有点象类中的this指针,它是java.lang.Object类的实例
    1 class getClass 返回此Object的类
    2 int hashCode() 返回此Object的hash码
    3 boolean equals(Object obj) 判断此Object是否与指定的Object对象相等
    4 void copy(Object obj) 把此Object拷贝到指定的Object对象中
    5 Object clone() 克隆此Object对象
    6 String toString() 把此Object对象转换成String类的对象
    7 void notify() 唤醒一个等待的线程
    8 void notifyAll() 唤醒所有等待的线程
    9 void wait(int timeout) 使一个线程处于等待直到timeout结束或被唤醒
    10 void wait() 使一个线程处于等待直到被唤醒
    11 void enterMonitor() 对Object加锁
    12 void exitMonitor() 对Object开锁

    6.application对象
    application对象实现了用户间数据的共享,可存放全局变量。它开始于服务器的启动,直到服务器的关闭,在此期间,此对象将一直存在;这样在用户的前后连接或不同用户之间的连接中,可以对此对象的同一属性进行操作;在任何地方对此对象属性的操作,都将影响到其他用户对此的访问。服务器的启动和关闭决定了application对象的生命。它是ServletContext类的实例。
    1 Object getAttribute(String name) 返回给定名的属性值
    2 Enumeration getAttributeNames() 返回所有可用属性名的枚举
    3 void setAttribute(String name,Object obj) 设定属性的属性值
    4 void removeAttribute(String name) 删除一属性及其属性值
    5 String getServerInfo() 返回JSP(SERVLET)引擎名及版本号
    6 String getRealPath(String path) 返回一虚拟路径的真实路径
    7 ServletContext getContext(String uripath) 返回指定WebApplication的application对象
    8 int getMajorVersion() 返回服务器支持的Servlet API的最大版本号
    9 int getMinorVersion() 返回服务器支持的Servlet API的最大版本号
    10 String getMimeType(String file) 返回指定文件的MIME类型
    11 URL getResource(String path) 返回指定资源(文件及目录)的URL路径
    12 InputStream getResourceAsStream(String path) 返回指定资源的输入流
    13 RequestDispatcher getRequestDispatcher(String uripath) 返回指定资源的RequestDispatcher对象
    14 Servlet getServlet(String name) 返回指定名的Servlet
    15 Enumeration getServlets() 返回所有Servlet的枚举
    16 Enumeration getServletNames() 返回所有Servlet名的枚举
    17 void log(String msg) 把指定消息写入Servlet的日志文件
    18 void log(Exception exception,String msg) 把指定异常的栈轨迹及错误消息写入Servlet的日志文件
    19 void log(String msg,Throwable throwable) 把栈轨迹及给出的Throwable异常的说明信息 写入Servlet的日志文件

    7.exception对象
    exception对象是一个例外对象,当一个页面在运行过程中发生了例外,就产生这个对象。如果一个JSP页面要应用此对象,就必须把isErrorPage设为true,否则无法编译。他实际上是java.lang.Throwable的对象
    1 String getMessage() 返回描述异常的消息
    2 String toString() 返回关于异常的简短描述消息
    3 void printStackTrace() 显示异常及其栈轨迹
    4 Throwable FillInStackTrace() 重写异常的执行栈轨迹

    8.pageContext对象
    pageContext对象提供了对JSP页面内所有的对象及名字空间的访问,也就是说他可以访问到本页所在的SESSION,也可以取本页面所在的application的某一属性值,他相当于页面中所有功能的集大成者,它的本 类名也叫pageContext。
    1 JspWriter getOut() 返回当前客户端响应被使用的JspWriter流(out)
    2 HttpSession getSession() 返回当前页中的HttpSession对象(session)
    3 Object getPage() 返回当前页的Object对象(page)
    4 ServletRequest getRequest() 返回当前页的ServletRequest对象(request)
    5 ServletResponse getResponse() 返回当前页的ServletResponse对象(response)
    6 Exception getException() 返回当前页的Exception对象(exception)
    7 ServletConfig getServletConfig() 返回当前页的ServletConfig对象(config)
    8 ServletContext getServletContext() 返回当前页的ServletContext对象(application)
    9 void setAttribute(String name,Object attribute) 设置属性及属性值
    10 void setAttribute(String name,Object obj,int scope) 在指定范围内设置属性及属性值
    11 public Object getAttribute(String name) 取属性的值
    12 Object getAttribute(String name,int scope) 在指定范围内取属性的值
    13 public Object findAttribute(String name) 寻找一属性,返回起属性值或NULL
    14 void removeAttribute(String name) 删除某属性
    15 void removeAttribute(String name,int scope) 在指定范围删除某属性
    16 int getAttributeScope(String name) 返回某属性的作用范围
    17 Enumeration getAttributeNamesInScope(int scope) 返回指定范围内可用的属性名枚举
    18 void release() 释放pageContext所占用的资源
    19 void forward(String relativeUrlPath) 使当前页面重导到另一页面
    20 void include(String relativeUrlPath) 在当前位置包含另一文件

    9.config对象
    config对象是在一个Servlet初始化时,JSP引擎向它传递信息用的,此信息包括Servlet初始化时所要用到的参数(通过属性名和属性值构成)以及服务器的有关信息(通过传递一个ServletContext对象)
    1 ServletContext getServletContext() 返回含有服务器相关信息的ServletContext对象
    2 String getInitParameter(String name) 返回初始化参数的值
    3 Enumeration getInitParameterNames() 返回Servlet初始化所需所有参数的枚举

    一 Servlet
    1、Servlet、ServletConfig:前者对应一个Servlet类,后者对应的是在web.xml中的配置信息
    (1)Servlet:
    init(ServletConfig):只再类加载并实例化后会被调用一次。
    service(ServletRequest,ServletResponse):会被调用多次
    destroy():只有一次
    getServletConfig()
    getServletInfo()
    (2)ServletCionfig
    getServletName()
    getInitParameter(String):---<servlet><init-param></init-param></servlet>
    getInitParameterNames():
    getServletContext
             注意ServletContext中也有getInitParameter(String),它对应的是:<context-param>中嵌套的元素
    这个<context-param>和<servlet>标签是同级别的,初始化的是整个应用程序

    2、两个类:
    GenericServlet/HttpServlet:
    GenericServlet实现Servlet和ServletConfig,HttpServlet继承自GenericServlet,而且它是和HTTP协议相关的。
    (1)GenericServlet:具有一个无参的init方法,方便子类的覆盖。
    (2)HttpServlet:具有两个service方法,并且有7个do方法。重写的时候一般重写do方法

    3、Servlet配置
    指的是web.xml中的信息,
    <servlet>
    <servlet-name></servlet-name>
    <servlet-class></servlet-class>
    <init-param></init-param>
    </servlet>
    //上面的配置对应的可以说就是ServletConfig中的东西
    <servlet-mapping>
    <servlet-name></servlet-name>
    <url-pattern></url-pattern>
    </servlet-mapping>

    4、WEB应用程序文件夹的结构
    --WEB-INF
    |
    |------------web.xml
    |------------lib文件夹      JAR文件
    |------------classes       类文件
    |------------tags          TagFile标签文件
    --静态资源与JSP文件

    5、请求和响应:
    (1)请求ServletRequest->HttpServletRequest
    getParameter(String)、
    getHeader()、
    getParameterValues(String)、用于复选框
    getParameterNames、
    getRemoteAddr、得到远程地址
    getLocalAddr、得到本地地址
    getLocale、得到本地化信息
    getSession/getCookies
    getRequestDispatcher

    (2)响应ServletResponse->HttpServletResponse
    sendRedirect
    sendError
    setHeader/addHeader/setIntHeader
    getWriter/getOutputStream :这两个方法绝对不可以同时被调用!!!!!
    setContentType/setCharacterEncoding

    6、请求的转发和响应的重定向之间的区别要搞清楚
    (1)转发:
    A、三种转发的方法:
    RequestDispatcher.forward/include
    <jsp:include>
    <jsp:forward> 等价于RequestDispatcher.forward();return;
    也就是说使用RequestDispatcher.forward()后,下面的代码仍然要执行,但是<jsp:forward>执行后,下面的代码就不执行了。

    pageContext.forward/include
    B、如何得到RequestDispatcher:
    request.getRequestDispatcher
    ServletContext.getRequestDispatcher

    RequestDispatcher的请求转发、Jsp页面中指令元素include、以及ServletContext中的getResourceAsStream(String path)
    (用来读出文件夹中的路径资源)都可以访问WEB-INF文件夹,
    比如为了防止用户访问一些受保护的页面(比如控制器Servlet),把它们放到WEB-INF中去,请求转发的时候可以去访问。
    有的时候需要读取WEB-INF中的一些配置资源,需要ServletContext中的getResourceAsStream方法。
    而且有的网站的各个页面具有相同的头图片和尾图片,这些东东就可以放到WEB-INF文件夹中用page元素的include指令来包含进来。

    注意如果使用ServletContext的RequestDispatcher,要从A应用程序转发请求到B,必须设定A的crossContext值为true!

    (2)重定向
    response.sendRedirect:比如在注册结束后转到下一个页面的时候,一定要使用重定向改变浏览器的URL地址!
    请求的转发不可能脱离Tomcat服务器的范围,如果想要脱离本服务器的话,只能通过响应重定向的方式。重定向要生成一个临时的响应,(响应
    一旦生成请求就结束了)浏览器接到这个临时的响应后不显示任何东西而是发送请求去找新的地址。


    二、JSP
    1、模板和元素:
    对于JSP来说,它是不可执行的,必须翻译成Servlet才能执行,必须要容器特殊处理的叫做元素,
    直接打印到输出流中去的是模板

    元素分为以下几种:
    1脚本元素
    2指令元素
    3动作元素

    1脚本元素:
    (1)脚本片断:翻译后原封不动的放到service方法里面
    (2)脚本声明:放在类里面,但是是在service方法外面,
    (3)脚本表达式:原封不动的放到out.print里面去,脚本表达式不可以加分号。

    内置对象绝对不可以在脚本声明中去用!因为它们的作用是在service方法里面的。

    2指令元素:
    Page指令:
    include指令<%@include file=""%>,包含的这个文件一定是按照纯文本的格式去读取,读取文件的时候就有编码的问题,这时候就是pageEncoding
    的设置问题了。
    taglib指令

    3页面乱码:pageEncoding/contentType
    前者设置jsp页面读取的形式,后者设置一个响应的报头,告诉浏览器以什么编码格式去显示
    如果没有设置pageEncoding而是只设置了contentType的话,那么pageEncoding要受contentType影响,反之亦然。
    也就是两者只设一者的话都会按照一个编码去显示!

    无论请求还是响应,递交的时候默认都是按照iso-8859-1去解码的。一定要保证读写和显示时候的编码都是一致的

    4 动作元素:
    <jsp:useBean>:主要作用:开放一个脚本变量并且向作用域里面存一个属性
    <jsp:setProperty>
    <jsp:getProperty>
    <jsp:forward>
    <jsp:include>
    <jsp:param>
    前六个比较重要!
    <jsp:invoke>
    <jsp:doBody>
    <jsp:plugin>
    <jsp:fallback>
    <jsp:params>

    三、过滤器和监听器
    1、Filter/FilterConfig
    init(FilterConfig)
    doFilter(ServletRequest,ServletResponse,FilterChain)
    destroy()

    <filter>
    <filter-name>
    <filter-class>
    </filter>

    <filter-mapping>
    <filter-name>
    <url-pattern>|<servlet-name>
    </filter-mapping>

    url-pattern相同的过滤器处于一个过滤器链上,执行的顺序完全按照web.xml中的先后顺序进行。

    2、监听器->ServletContext/HttpSession/ServletRequest
    (1)生命周期
    ServletContextListener:初始化(从BBS中读取全部讨论区并存储在应用程序的作用域中或是将全部封杀的IP读出来存储)和销毁
    HttpSessionListener/HttpSessionActivationListener
    ServletRequestListener
    (2)属性的增删改
    一般是AttributeListener去作的
    <listener>
    <listener-class>包名.类名</listener-class>
    </listener>

    四、四个作用域和九个内置对象
    1、页面作用域对应的是pageContext,而不是page
    2、内置对象:
    pageContext/request/session/application
    out->是JspWriter的实例,它有缓存,而PrintWriter没有缓存,在页面结束的时候一定会通过响应生成PrintWriter去写出缓存的内容!
    如果缓存满了的话,要看:如果<%@page autoFlush="true"%>,则会自动刷新,如果是false,会抛出异常的!response.getWriter()
    返回的是PrintWriter,不是JspWriter。
    exception->isErrorPage=true的时候才有用
    session:两种跟踪机制:一种是通过客户端的cookie存储,另外一种是将sessionId存在服务器端,
    config->ServletConfig
    page->Object->this
    response:

    五、EL表达式语言:
    ${}
    1、常量、函数、变量的表达式
    2、变量是存储在某一作用域中的同名的属性值
    3、函数是在tld文件中声明的引用一个类的公共且静态的方法
    4、11个内置对象
    (1)作用域:pageScope/requestScope/sessionScope/applicationScope
    (2)和页面通信:pageContext
    (3)请求参数的:param/paramValues
    (4)报头:header/headerValues
    (5)Cookies/initParam ---<context-param>
    5、点操作符和[]操作符和empty
    对javaBean,是调用getter方法,
    对map,是调用get(Object)方法,
    对作用域,是调用getAttribute(String)
    对pageContext,也是调用getter方法,可以得到所有的内置对象
    对param,是去取得参数
    对header,是去拿报头
    对initParam,是去拿初始化参数
    []操作符主要是针对数组


    六 自定义标签
    1、Tag接口(传统接口系列)

    JspTag
    (1)Tag : doStartTag(返回值是SKIP_BODY/EVAL_BODY_INCLUDE)/doEndTag(返回SKIP_PAGE或是EVAL_PAGE)
    (2)IterationTag->TagSupport
    doAfterBody:EVAL_BODY_AGAIN/SKIP_BODY
    (3)BodyTag->BodyTagSupport
    doInitBody
    doAfterBody:EVAL_BODY_BUFFERED->BodyContent:pageContext里面有一个pushBody,用来将体包起来,这时候再用getOut方法得到
    的不再是JspWriter了,而是bodyContent了。BodyContent的生成是调用了pushBody方法,这里面又涉及一个getEnclosingWriter(),它
    是BodyContent的一个方法,可以得到它里面包的输出流JspWriter,但是要注意要是进行了一次以上的pushBody但是没有进行popBody的话,那么
    拿出来的就是里面包的BodyContent了,但是由于BodyContent是JspWriter的子类,所以也是符合方法声明的


    2、SimpleTag接口(简单接口系列)
    setJspContext()
    setParent()
    setJspBody(JspFragment) 注意JspFragment是标签体,里面绝对不可以有脚本元素的!注册的时候content元素中要么设置为empty,
    要么设置为scriptless,也就是简单标签的标签体是不可以有脚本元素的!
    doTag()

    JspFragment.invoke(null):直接弄到输出流中去
    JspFragment.invoke(StringWriter out):将输出流弄到StringWriter中去,然后调用out.toString可以将体变成字符串的形式

    3、Tag File标签文件
    .tag
    可以放在/WEB-INF/tags里面或者/META-INF/tags里面也可以
    至于标签文件的tld可以放在WEB-INF中也可以放在META-INF中去,都是没有问题的
    标签文件描述的是标签处理类,jsp描述的是Servlet,二者都是不可运行的,注意标签文件中有<%@tag%>,里面可以设置不少标签的属性
    还有<%@attribute %>,<%@taglib%>和<%@include%>也可以用,属性可以是JspFragment

    比如:<my:first>
       <jsp:attribute>
       </jsp:attribute>
       <jsp:body>
       </jsp:body>
    </my:first>

    <jsp:invoke name=var>运算结果就放在了var里面
    <jsp:doBody> 处理体

    七、JSTL
    一共5个库,我们说了core、sql、i18n、fn标记库(就是表达式语言的函数库)
    我们没有讲xml的标签库

    八、国际化与汉字的编码
    ResourceBundle
    两种形式:
    1、ListResourceBundle
    2、写一个Properties文件,里面有一个nativetoacsii命令进行,参照前面的笔记转码
    基名和扩展名:起名字都是要有标准的
    Locale:本地化对象
    getBundle方法可以通过传入基名和本地化对象来处理
    <fmt:bundle>
    <fmt:setBundle>
    <fmt:message>
    汉字编码:GB2312/GB13000(GBK)/GB18030
    Unicode UCS/UTF-8

    JSP的九大内置对象

    内置对象(又叫隐含对象,有9个内置对象):不需要预先声明就可以在脚本代码和表达式中随意使用

    1-out:

    javax.servlet.jsp.JspWriter类型,代表输出流的对象。作用域为page(页面执行期)

    request:javax.servlet.ServletRequest的子类型,此对象封装了由WEB浏览器或其它客户端生成地HTTP

    请求的细节(参数,属性,头标和数据)。作用域为request(用户请求期)。
    方法由可查看API

    response:javax.servlet.ServletResponse的子类型,此对象封装了返回到HTTP客户端的输出,向页面

    作者提供设置响应头标和状态码的方式。经常用来设置HTTP标题,添加cookie,设置响应内容的类型和状

    态,发送HTTP重定向和编码URL。作用域为page(页面执行期)。


    pageContext:javax.servlet.jsp.PageContext(抽象类)类型,作用域为page(页面执行期)。此对象

    提供所有四个作用域层次的属性查询和修改能力,它也提供了转发请求到其它资源和包含其他资源的方法


    该对象的方法都是抽象方法

    session:javax.servlet.http.HttpSession类型,主要用于跟踪对话。作用域session(会话期—)。

    HttpSession是一个类似哈希表的与单一WEB浏览器会话相关的对象,它存在于HTTP请求之间,可以存储任

    何类型的命名对象。
    如果不需要在请求之间跟踪会话对象,可以通过在page指令中指定session="false"
    需要记住的是pageContext对象也可以与session.getAttribute(),session.setAttribute()一样的方式取

    得并设置会话属性。

    application:javax.servlet.ServletContext类型,servlet的环境通过调用getServletConfig

    ().getContext()方法获得。作用域是application(整个程序运行期)。它提供了关于服务器版本,应用

    级初始化参数和应用内资源绝对路径,注册信息的方式


    config:javax.servlet.ServletConfig,作用域为page(页面执行期)

    exception:java.lang.Throwable,通过JSP错误页面中一个catch块已经益出但没有捕获的

    java.lang.Throwable的任意实例,传向了errorPage的URI。作用域为page(页面执行期)。注意

    exception只有在page指令中具有属性isErrorPage="true"时才有效。

    page:java.lang.Object类型,指向页面自身的方式。作用域为page(页面执行期

    JSP中九大内置对象为:
    request            请求对象                 类型 javax.servlet.ServletRequest        作用域 Request
    response          响应对象                   类型 javax.servlet.SrvletResponse       作用域 Page
    pageContext    页面上下文对象       类型 javax.servlet.jsp.PageContext      作用域    Page
    session            会话对象                   类型 javax.servlet.http.HttpSession       作用域    Session
    application       应用程序对象          类型 javax.servlet.ServletContext          作用域    Application
    out                   输出对象                   类型 javax.servlet.jsp.JspWriter             作用域    Page
    config              配置对象                  类型 javax.servlet.ServletConfig            作用域    Page
    page               页面对象                  类型 javax.lang.Object                            作用域    Page
    exception        例外对象                 类型 javax.lang.Throwable                     作用域    page

    request 对象代表的是来自客户端的请求,例如我们在FORM表单中填写的信息等,是最常用的对象

    常用的方法有:getParameter、getParameterNames 和getParameterValues 通过调用这几个方法来获取请求对象中所包含的参数的值。

    response 对象代表的是对客户端的响应,也就是说可以通过response 对象来组织发送到客户端的数据。但是由于组织方式比较底层,所以不建议普通读者使用,需要向客户端发送文字时直接使用

    pageContext 对象直译时可以称作“页面上下文”对象,代表的是当前页面运行的一些属性

    常用的方法有 :findAttribute、getAttribute、getAttributesScope 和getAttributeNamesInScope
    一般情况下pageContext对象用到得也不是很多,只有在项目所面临的情况比较复杂的情况下,才会利用到页面属性来辅助处理。

    session    对象代表服务器与客户端所建立的会话,当需要在不同的JSP页面中保留客户信息的情况下使用,比如在线购物、客户轨迹跟踪等。“session” 对象建立在cookie的基础上,所以使用时应注意判断一下客户端是否打开了cookie。常用的方法包括getId、 getValue、 getValueNames和putValue等。

    概要
    HTTP是无状态(stateless)协议;
    Web Server 对每一个客户端请求都没有历史记忆;
    Session用来保存客户端状态信息;
    由Web Server 写入;
    存于客户端;
    客户端的每次访问都把上次的session记录传递给Web Server;
    Web Server读取客户端提交的session来获取客户端的状态信息

    application 对象负责提供应用程序在服务器中运行时的一些全局信息,常用的方法有getMimeType和getRealPath等。

    out 对象代表了向客户端发送数据的对象,与“response” 对象不同,通过“out” 对象发送的内容将是浏览器需要显示的内容,是文本一级的,可以通过“out” 对象直接向客户端写一个由程序动态生成HTML文件。常用的方法除了pirnt和println之外,还包括clear、clearBuffer、flush、getBufferSize和getRemaining,这是因为“out” 对象内部包含了一个缓冲区,所以需要一些对缓冲区进行操作的方法

    “config” 对象提供一些配置信息,常用的方法有getInitParameter和getInitParameterNames,以获得Servlet初始化时的参数。

    “page” 对象代表了正在运行的由JSP文件产生的类对象,不建议一般读者使用。

    “exception” 对象则代表了JSP文件运行时所产生的例外对象,此对象不能在一般JSP文件中直接使用,而只能在使用了“<%@ page isErrorPage="true "%>”的JSP文件中使用 


    JSP的四个作用域

    何为作用域
    先让我们看看效果:


    大概流程是这样的,我们访问04-01/index.jsp的时候,分别对pageContext, request, session,

    application四个作用域中的变量进行累加。(当然先判断这个变量是不是存在,如果变量不存在,则要

    把变量初始化成1。)计算完成后就从index.jsp执行forward跳转到test.jsp。在test.jsp里再进行一次

    累加,然后显示出这四个整数来。

    从显示的结果来看,我们可以直观的得出结论:

    page里的变量没法从index.jsp传递到test.jsp。只要页面跳转了,它们就不见了。

    request里的变量可以跨越forward前后的两页。但是只要刷新页面,它们就重新计算了。

    session和application里的变量一直在累加,开始还看不出区别,只要关闭浏览器,再次重启浏览器访问

    这页,session里的变量就重新计算了。

    application里的变量一直在累加,除非你重启tomcat,否则它会一直变大。

    而作用域规定的是变量的有效期限。

    如果把变量放到pageContext里,就说明它的作用域是page,它的有效范围只在当前jsp页面里。

    从把变量放到pageContext开始,到jsp页面结束,你都可以使用这个变量。

    如果把变量放到request里,就说明它的作用域是request,它的有效范围是当前请求周期。

    所谓请求周期,就是指从http请求发起,到服务器处理结束,返回响应的整个过程。在这个过程中可能使

    用forward的方式跳转了多个jsp页面,在这些页面里你都可以使用这个变量。

    如果把变量放到session里,就说明它的作用域是session,它的有效范围是当前会话。

    所谓当前会话,就是指从用户打开浏览器开始,到用户关闭浏览器这中间的过程。这个过程可能包含多个

    请求响应。也就是说,只要用户不关浏览器,服务器就有办法知道这些请求是一个人发起的,整个过程被

    称为一个会话(session),而放到会话中的变量,就可以在当前会话的所有请求里使用。

    如果把变量放到application里,就说明它的作用域是application,它的有效范围是整个应用。

    整个应用是指从应用启动,到应用结束。我们没有说“从服务器启动,到服务器关闭”,是因为一个服务

    器可能部署多个应用,当然你关闭了服务器,就会把上面所有的应用都关闭了。

    application作用域里的变量,它们的存活时间是最长的,如果不进行手工删除,它们就一直可以使用。

    与上述三个不同的是,application里的变量可以被所有用户共用。如果用户甲的操作修改了application

    中的变量,用户乙访问时得到的是修改后的值。这在其他scope中都是不会发生的,page, request,

    session都是完全隔离的,无论如何修改都不会影响其他人的数据。

    我们使用public Object getAttribute(String name)获得变量值,使用public void setAttribute

    (String name, Object value)将变量值保存到对应作用域中。举个pageContext的例子就是:

    // page
    Integer countPage = (Integer) pageContext.getAttribute("countPage");
    if (countPage == null) {
        pageContext.setAttribute("countPage", 1);
    } else {
        pageContext.setAttribute("countPage", countPage + 1);
    }
    这里先从pageContext中取出名为countPage的整数,因为返回的都是java.lang.Object类型,所以需要强

    制转换成我们需要的整形。这里取得的变量如果不存在就会返回null,通过判断countPage == null来辨

    别变量是否存在,如果不存在就设置为1,如果存在就进行累加,最后使用setAttribute()方法将修改后

    的变量值放入pageContext。

    将其中的pageContext换成request, session, application就可以操作其他三个作用域中的变量

  • 相关阅读:
    NOP源码分析六--实体、数据的分层与处理。
    NOP源码分析七---继续
    NOP源码分析 八---set的存储
    Nop 源码分析四 任务系统
    NOP源码分析五,文件位置等详细内容,感冒真难受,嗓子痒又疼。。
    1
    mobx
    ts随笔
    13.vue-vuex
    13.vue-axios
  • 原文地址:https://www.cnblogs.com/hy928302776/p/3078181.html
Copyright © 2011-2022 走看看