JavaWeb(三)JSP之3个指令、6个动作、9个内置对象和4大作用域

前言

  前面大概介绍了什么是JSP,今天我给大家介绍一下JSP的三个指令、6个动作以及它的9大内置对象。接下来我们就直接进入正题

一、JSP的3个指令

JSP指令(directive)是为JSP引擎而设计的,它们并不直接产生任何可见输出,而只是告诉引擎如何处理JSP页面中的其余部分

指令用来申明JSP页面的一些属性,比如编码方式,文档类型。我们在servlet中也会申明我们使用的编码方式和响应的文档类型的,而JSP就是用指令来申明。上面我们也说到了一条指令,也就是page指令。

JSP指令格式:<%@ directive {attribute=value}* %>(<%@ 指令名称 属性1=“属性值1” 属性2=“属性值2”。。。%>)

  分析:  

    directive:指令名称,例如page指令

    attribute=value:紧跟指令名称后面的就是各种属性,以键值对的形式书写

    *:代表后面能跟0个或多个属性。

1.1、page指令(用来声明JSP页面的属性等)

  <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>  page指令,后面跟着三个属性,分别是language、contentType、pageEncoding。

  这只是其中的几个属性,并没有写全,page指令允许的属性如下表所示: 

        属性名称        取值范围          描述

        language        java      解释该JSP文件时采用的语言,一般为java语言,默认为java

        extends        任何类的全名   编译该JSP文件时继承哪个类,JSP为Servlet,因此当指明继承普通类时需要实现Servlet的init、destroy等方法

        import         任何包名、类名  引入该JSP中用到的类、包等,import是唯一可以声明多次的page指令属性,一个import可以引用uogelei,中间用英文逗号隔开,

                              如<%@ page import="java.util.List,java.util.ArrayList"%>

        session        true、false    该JSP内是否内置Session对象,如果为true,则内置Session对象,可直接使用,否则反之,默认为true

        autoFlush       true,false    是否运行缓存,如果为true,则使用out.println()等方法输出的字符串并不是立刻到达客户端服务器的,而是暂时存到缓存里,缓存满

                             了或者程序执行完毕或者执行out.flush()操作时才到客户端,默认为true。

        buffer        none或者数字KB   指定缓存大小,当autoFlush设为true时有效,例如<%@ page buffer=10kb%>

        isThreadSafe      true,false    是否线程安全,如果为true,则运行多个线程同时运行该jsp程序,否则只运行一个线程,其余线程等待,默认为false

        isErrorPage      true,false     指定该页面是否为错误显示页面,如果为true,则该JSP内置有一个Exception对象exception,可直接使用,否则没有,默认为false

        errorPage     某个JSP页面的相对路径  指明一个错误页面,如果该JSP程序抛出一个未捕捉的异常,则转到errorPage指定的页面,errorPage指定的页面通常

                               isErrorPage属性为true,且内置的exception对象为未捕捉的异常

        contentType     有效的文档类型    客户端浏览器根据该属性判断文档类型,例如 HTML格式为text/html、纯文本格式为text/plain、JPG图像为image/jpeg、GIF图像为image/gif、

                              WORD文档为application/msword,该属性常跟着charset设置编码一起,作用是通知服务器和浏览器都使用同一个码表

        info          任意字符串      指明JSP的信息,该信息可以通过Servlet.getServletInfo()方法获取到

    trimDirective Whitespaces    true、false     是否去掉指令前后的空白字符,默认为false

        pageEncoding    UTF-8,ISO-8859-1等    指定一张码表来对该JSP页面进行编码

1.2、include指令

  比较简单,只有一种形式 <%@ include file="relativeURL"%>  relativeURL:本应用程序内另一个JSP文件或者HTML文件的路径,例如,网址内所有页面均有一个统一风格的导航栏和页脚版权,那么就可以使用该指令将其包含进来。

  特点:include指令会将包含页面的源代码添加到使用include指令的页面中来,然后编译成class文件,而等下会讲到的一个JSP行为,<jsp:include page="relativeURL">作用跟include指令一样,

      但是不同的是,include行为是运行时单独执行包含页面,然后把执行的结果包含到本页面来,属于先运行后包含。  

  注意:  

    静态包含:把其它资源包含到当前页面中。
      <%@ include file="/include/header.jsp" %>
    动态包含:
      <jsp:include page="/include/header.jsp"></jsp:include>

    两者的区别:翻译的时间段不同
      前者:在翻译时就把两个文件合并
      后者:不会合并文件,当代码执行到include时,才包含另一个文件的内容。

    原则:能用静的就不用动的。

1.3、taglib指令

  JSP支持标签技术,后面会讲到标签的用法,jstl标签库的使用等,

  作用:用来指明JSP页面内使用的JSP标签库,taglib指令有两个属性,uri为类库的地址,prefix为标签的前缀

  <%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>

二、JSP的6个动作(行为)

前面讲了JSP语法,介绍了JSP页面中的内容有哪些,分别有什么作用,就两个东西,模块数据和元素。其中元素有包括脚本,指令,标签,脚本就是JSP中嵌入java代码,指令作用就是申明页面的属性,

那标签是干嘛的,标签分为JSP自带内置的标签,和通过taglib指令来使用JSP标签库,或者自定义标签。现在我们先来讲一些JSP内置的标签。 

JSP内置的标签就被称为JSP行为(JSP Actions)。只要书写很少的标记代码就能使用JSP提供的丰富功能,JSP行为其实是对常用的JSP功能的抽象与封装,可以取代jsp脚本,让JSP中就少一些嵌入java代码的地方

简单的说就是使用标签的形式来表示一段java代码 

格式:<jsp:elements {attribute="value"}* />  

  分析:

     jsp:标签的前缀,说明是jsp内置的标签,

     elements:行为的名称,

     attribute=value:使用键值对来编写属性

     *:能指定0个或多个属性对

2.1、<jsp:include />行为(动态包含)

  <jsp:include page="/include/header.jsp"></jsp:include>

  include行为用于运行时包含某个文件,如果被包含的文件为JSP程序,则先会执行JSP程序,然后在把执行的结果包含进来。 

  作用是跟include指令一样的,唯一的区别就在于,include指令是将被包含的文件的源码加入到了本JSP程序中,然后在进行编译,属于静态包含,而include行为只是将被包含的文件的运行结果包含进自己。属于动态包含。

    

2.2、Java bean行为

  是一组与Java Bean 相关的行为,包括useBean行为、setProperty行为、getProperty行为    

  Java Bean就是普通的Java类,也被称为POJO,只有私有的属性与对应的getter方法和setter方法,注意其中当私有的属性为boolean类型时,习惯上一般把getter方法写成isXxx();而不是getXxx();  

  1)userBean行为

    <jsp:useBean id="beanObject" class="className" scope="Value">  作用:在jsp中定义一个java bean对象

    分析:   

      id:指明Java Bean对象的名称,JSP中可以使用该名称引用该Java Bean对象,相当于给new出来的对象取一个变量名,

      class:Java Bean类的全名

      scope:该java bean对象的作用范围,可以写的就四个,也就是JSP的四大作用域,page、request、session、application

        page:只能在当前JSP页面使用,如果不在JSP页面,那么就会失效

        request:这个前面学过,A页面请求转发到B页面,那么使用的是同一个request,那么A,B页面都算是request的作用域,也就是通过请求转发的页面都是其作用域

        session:该作用域在一个web项目下任何位置应该读访问的到,只要cookie不关闭,并且cookie设置  的访问路径为"/",

          application:其实就是Servlet中的servletContext,服务器下的所有项目都能访问到。

  2)setProperty行为

    <jsp:setProperty name="beanName" property="propertyName" value="">

    分析:

      对Java Bean对象进行属性的设置

      name:java bean对象的名称,也就是在useBean行为中的id

      property:对象中的属性名,

      value:要对其属性进行赋值的值

  3)getProperty行为

    <jsp:getProperty name="beanName" property="propertyName" />

    分析:            

      获取JavaBean对象的某个属性值

      name:java bean 对象的名称,也就是在useBean行为中的id

      property:对象的属性名

  举例:javabean:User.java、NewFile.jsp

  User.java

package a;

public class User {
    private int id;
    private String username;
    private String password;
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username = username;
    }
    public String getPassword() {
        return password;
    }
    public void setPassword(String password) {
        this.password = password;
    }
    
    
}
User

  NewFile.jsp

<body>
    <!-- 创建一个新的javabean对象user,会先判断在page作用域内是否有叫user对象的javabean,如果有则取它,如果没有则创建新的javabean对象  -->
    <jsp:useBean id="user" class="a.User" scope="page"></jsp:useBean>
    <!-- 对javabean对象的username进行赋值 -->
    <jsp:setProperty property="username" name="user" value="faker"/>
    <!-- 获取javabean对象的username属性 -->
    <jsp:getProperty property="username" name="user"/>
</body>
NewFile.jsp

  可以查看NewFile.jsp变为servlet后的源代码,看看我们写的javabean行为会被转换为何种语句:

  

  这里出现了一个JSP九大内置对象中的一个,pageContext。现在简单提一句,pageContext就是JSP页面的管理者(上下文),其中的getAttribute(name,scope)方法是获取指定作用域中的数据的,

  如果getAttribute(name)方法的话,默认是对page作用域进行操作,findAttribute(name)依次从page、request、session、application获得内容。

  在第一个红框中,就代表中我们的useBean行为,其中进行了一次判断,就是如果在page作用域中找不到user这个对象,那么就创建一个新的,否则就使用找到的这个user对象,

  第二个红框中,代表着我们的setProperty行为,先找到user对象,然后在对其属性进行赋值

  第三个红框中,代表着我们的getProperty行为,也是先找到user对象,然后在获取其属性的值。

  注意:对于javabean行为来说,有一个特点的地方,就是当请求过来的参数对应javabean的属性时,可以为其一次性设置所有的值

      <jsp:setProperty name="user" property="*" /> //设置user的所有属性,属性值从request中自动取得,*代表所有属性。

2.3、<jsp:forward />行为

  实现请求转发功能,Servlet中通过request.getRequestDispatcher("someServlet").forward(request,response);而在JSP中也能够实现相同的功能,只不过用的是<jsp:forward />行为,实际上forward行为就是对其进行了封装。  

  格式:

        <jsp:forward page="someServlet">
           <jsp:param name="param1" value="value1"/>
           <jsp:param name="param2" value="value2"/>
        </jsp:forward>

  分析:page:需要跳转到的页面或者servlet、<jsp:param/>参数行为,带一些参数过去,name、value是以键值对的形式带过去的     

  举例:

    在NewFile.jsp中

    

    在ForwardTestServlet中

    

    访问:http://localhost:8080/Web_Jsp/NewFile.jsp

    

    浏览器地址栏没有改变,说明是请求转发

2.4、<jsp:directive/>行为

  directive行为,就相当于JSP指令,比如<jsp:directive.page/>相当于<%@ page %>指令,等等其它指令是一样的书写格式。

总结:

  在我们开发中,经常要用到的6个行为是

    <jsp:include > 动态包含
    <jsp:forward> 请求转发
    <jsp:param> 设置请求参数

    <jsp:useBean> 创建一个对象
    <jsp:setProperty> 给指定的对象属性赋值
    <jsp:getProperty> 取出指定对象的属性值

三、JSP隐藏的九大内置对象

我们知道JSP中的内容就只有两种,模版数据和元素,元素就包括了指令,脚本,标签(行为)脚本会慢慢被标签全部代替,也就是说JSP中基本上不会嵌入Java代码,但是我们也知道JSP会转换为servlet,

在Servlet中,输出数据时,都需要通过response.getWrite();但是在JSP中,直接使用out对象进行输出,为什么呢?这就是因为out为JSP的一个隐藏对象,JSP中内置了9个隐藏对象,使得JSP比Servlet使用起来更简单,更方便。

3.1、九大内置对象概述

   

  分析:

    request:请求对象,  类型:httpServletRequest

    response:响应对象  类型:httpServletResponse

    session:表示一次会话,在服务器端记录用户状信息的技术

    application:标识web应用上下文,类型:ServletContext,详情就看Servlet中的ServletContext的使用

    exception 表示发生异常对象,类型 Throwable,在上面我们介绍page指令中的一个errorPage属性时就有说到他

    page:page对象代表当前JSP页面,是当前JSP编译后的Servlet类的对象。相当于this。

    config:标识Servlet配置,类型:ServletConfig,api跟Servlet中的ServletConfig对象是一样的,能获取该servlet的一些配置信息,能够获取ServletContext

    out:输出响应体 类型:JspWriter

    pageContext:表示 jsp页面上下文(jsp管理者) 类型:PageContext

    注意:标记了红色的对象就是JSP独有的,其他的都是Servlet中的老东西。

  

  在这个由jsp转换为servlet的文件中,只能看到8个内置对象,少了exception对象,因为我们在将page指令时,说过一个isErrorPage属性,默认是false,被关闭了,所以其中并没有exception对象。

3.2、pageContext(重要)

  这个功能就比较强大了,基本上什么他都有,因为是它是JSP页面的管理者(上下文),所以JSP中的内置对象呀,它统统能够获得,下面介绍它的api:

  1)获得其它八大内置对象 getXxx()

    在普通类中可以通过PageContext获取其他JSP隐式对象。自定义标签时就使用。

    pageContext.getOut();  //获得out对象

    pageContext.getApplication();  //获得application对象

    等等....

  2)对作用域的属性进行操作(四大作用域)

    对默认作用域的属性进行操作。page

    Object getAttribute(String name);  //获得page作用域数据

    void setAttribute(String name,Object o);  //给page作用域设置内容

    void removeAttribute(String name);  //给page作用域移除内容

  3)对指定作用域的属性进行操作

    Object getAttribute(String name,int Scope);  //获得 指定作用域中的数据

    void setAttribute(String name,Object o,int Scope);  //给指定作用域设置内容

    void removeAttribute(String name,int Scope);     // 移除指定作用域的内容(page/request/session/application)

  4)提供作用域常量

    PageContext.PAGE_SCOPE  page

    PageContext.REQUEST_SCOPE  request      

    PageContext.SESSION_SCOPE  response

    PageContext.APPLICATION_SCOPE  application

  5)一次获得指定名称内容

    page中最厉害的方法是:

      findAttribute(String name);  //自动从page request session application依次查找,找到了就取值,结束查找。

  实例:

    在1.jsp中:

    

    在2.jsp中

    

  6)提供了的简易方法  

    pageContext.forward("2.jsp");
    pageContext.include("2.jsp");

3.3、out对象

  类型:JspWriter

  jsp 输出底层使用 response.getWriter();什么意思呢?这里就要讲解一下JSP缓存和Servlet缓存了,输出的过程是这样的

  

  JSP页面转换为Servlet后,使用的out对象是JspWriter类型的,所以是会先将要发送的数据存入JSP输出缓存中,然后,等JSP输出缓存满了在自动刷新到servlet输出缓存,

    等serlvet输出缓存满了,或者程序结束了,就会将其输出到浏览器上。除非手动out.flush()。

  验证servlet输出缓存和JSP输出缓存和我们上面所说的是正确:

    

     结果:

       

    分析:

      如果按没有jsp缓存和servlet缓存的话,输出的结果应该是aaaabbbbcccc,但是输出的却是bbbbaaaacccc,为什么呢?按照我们上面所说的原理进行分析,out对象是先将其输出到JSP缓存中,所以aaaa加入了jsp缓存,

      而response.getWriter().print("bbbb")是直接将bbbb输出到servlet缓存中,然后又使用out对象将cccc输出到jsp缓存,到程序结束,servlet缓存中有bbbb,然后jsp会将缓存中的内容就刷新到servlet缓存中,

      serlvet就是bbbbaaaacccc了,然后到浏览器也就得到我们的输出结果了。如果在12行将注释去掉,那么输出的结果又会是什么呢?答案就是aaaabbbbcccc,过程自行分析。

      

3.4、config对象

  类型:ServletConfig

  能够获取servlet的初始化参数,获取servletContext对象,获取servletName。

  这个我在servlet中详细的讲解了,可以去查看!

3.5、exception异常对象

  包含了异常的信息

  使用它,必须结合page指令中的isErrorPage属性和errorPage属性

  exception.jsp  抛异常的一个NullPointException,并且跳转到error.jsp错误显示页面  errorPage属性的意思是如果发生未捕捉到的异常,将会跳转到error.jsp页面

  举例:

    

    error.jsp  isErrorPage属性说明该页面是一个错误显示页面,则可以使用exception对象

    

  访问:访问http://localhost:8080/Web_Jsp/exception.jsp

    

总结:九大内置对象和servlet中对象的关系  

  page就是jsp转换为servletservlet对象本身,也就是this

  config -- Servlet中的servletConfig

  application -- Servlet中的ServletContext

  request  -- Servlet中的request

  response  -- Servlet中的response

  session  -- Servlet中的session    

  out  -- JspWriter

  exception  -- 异常对象

  pageContext  -- 表示 jsp页面上下文(jsp管理者) 类型:PageContext

  其中pageContext是最厉害的,因为它可以得到其他8个内置对象  

四、JSP四大作用域

这四大作用域,其实就是其九大内置对象中的四个,为什么说他们也是JSP的四大作用域呢?

  因为这四个对象都能存储数据,比如request.setAttribute()注意和request.setParameter()区分开来,一个是存储在域中的、一个是请求参数,session.setAttribute()、application其实就是SerlvetContext,自然也有setAttribute()方法。

  而page作用域的操作就需要依靠pageContext对象来进行了。在上面我们也有提到JSP的四大作用域。

1)page作用域

  代表变量只能在当前页面上生效

2)request作用域

  代表变量能在一次请求中生效,一次请求可能包含一个页面,也可能包含多个页面,比如页面A请求转发到页面B。

3)session作用域

  代表变量能在一次会话中生效,基本上就是能在web项目下都有效,session的使用也跟cookie有很大的关系。一般来说,只要浏览器不关闭,cookie就会一直生效,cookie生效,session的使用就不会受到影响

4)application作用域

  代表变量能一个应用下(多个会话),在服务器下的多个项目之间都能够使用。比如baidu、wenku等共享帐号。

 

  

喜欢就点个“推荐”哦!   

posted @ 2017-10-08 23:07  华仔Coding  阅读(9234)  评论(2编辑  收藏  举报
levels of contents