java基础学习:JavaWeb之JSP

其余更多java基础文章:
java基础学习(目录)html

本章内容较多,所有认真看完可能须要一小时以上,建议边看边作笔记,不然容易混乱java


1、JSP概述

1.一、JSP结构

网络服务器须要一个JSP引擎,也就是一个容器来处理JSP页面。容器负责截获对JSP页面的请求。内嵌JSP容器的Apache支持JSP开发。
JSP容器与Web服务器协同合做,为JSP的正常运行提供必要的运行环境和其余服务,而且可以正确识别专属于JSP网页的特殊元素。
下图显示了JSP容器和JSP文件在Web应用中所处的位置。web

1.二、JSP运行流程

如下步骤代表了Web服务器是如何使用JSP来建立网页的:编程

  1. 就像其余普通的网页同样,您的浏览器发送一个HTTP请求给服务器。Web服务器识别出这是一个对JSP网页的请求,而且将该请求传递给JSP引擎。经过使用URL或者.jsp文件来完成。
  2. JSP引擎从磁盘中载入JSP文件,而后将它们转化为servlet。这种转化只是简单地将全部模板文本改用println()语句,而且将全部的JSP元素转化成Java代码。
  3. JSP引擎将servlet编译成可执行类,而且将原始请求传递给servlet引擎。
  4. Web服务器的某组件将会调用servlet引擎,而后载入并执行servlet类。在执行过程当中,servlet产生HTML格式的输出并将其内嵌于HTTP response中上交给Web服务器。Web服务器以静态HTML网页的形式将HTTP response返回到您的浏览器中。 最终,Web浏览器处理HTTP response中动态产生的HTML网页,就好像在处理静态网页同样。

以上说起到的步骤能够用下图来表示: api

通常状况下,JSP引擎会检查JSP文件对应的servlet是否已经存在,而且检查JSP文件的修改日期是否早于servlet。若是JSP文件的修改日期早于对应的servlet,那么容器就能够肯定JSP文件没有被修改过而且servlet有效。这使得整个流程与其余脚本语言(好比PHP)相比要高效快捷一些。浏览器

Jsp 生成java源码,默认第一次生成,以后直接执行,除非内容修改,具体点说,因为JSP只会在客户端第一次请求的时候被编译,所以第一次请求JSP时会感受比较慢,而以后的请求由于不会编译JSP,因此速度就快多了。
若是将Tomcat保存的JSP编译后的class文件删除,Tomcat也会从新编译JSP。在开发Web程序的时候常常须要修改JSP, Tomcat可以自动检测到JSP程序的改动,若是检测到JSP源代码发生了改动,Tomcat会在下次客户端请求JSP时从新编译JSP,而不须要重启Tomcat,这种自动检测功能默认是开启的,检测改动会消耗少许的时间,在部署web应用程序的时候能够在web.xml中将它关掉。
这也就是为何咱们可以在jsp页面直接修改内容,而不用从新启动服务器的缘由。缓存

总的来讲,JSP网页就是用另外一种方式来编写servlet而不用成为Java编程高手。除了解释阶段外,JSP网页几乎能够被当成一个普通的servlet来对待。安全

1.三、JSP生命周期

理解JSP底层功能的关键就是去理解它们所遵照的生命周期。JSP生命周期就是从建立到销毁的整个过程,相似于servlet生命周期,区别在于JSP生命周期还包括将JSP文件编译成servlet。
如下是JSP生命周期中所走过的几个阶段:
编译阶段:
servlet容器编译servlet源文件,生成servlet类
初始化阶段:
加载与JSP对应的servlet类,建立其实例,并调用它的初始化方法
执行阶段:
调用与JSP对应的servlet实例的服务方法
销毁阶段:
调用与JSP对应的servlet实例的销毁方法,而后销毁servlet实例
很明显,JSP生命周期的四个主要阶段和servlet生命周期很是类似,下面给出图示:bash

2、JSP语法

2.一、jsp脚本

  1. 使用<% 编写java代码 %>,中间java代码必须遵循Java语法
          
    来看看,jsp变为servlet时的代码是如何编写的

      

2. 使用<%=xxx %>来输出结果

      

使用<%=result %>来输出结果,servlet中就会将其转换为out.print(result)进行输出。输出各类类型数据:int、double、boolean、String、Object等。服务器

      

  1. 注释   

<%-- --%>:jsp注释 <!-- -->:这个注释,会发送到浏览器端的源码中显示 注释分别在servlet中如何显示:

        

在servlet中

        

总结:JSP注释不会在servlet文件中显示,而java注释则会,但其全部的注释到了浏览器端,都不会出如今源码中,只有这个注释会到浏览器的网页源码中去。

  1. JSP中申明方法与属性(全局变量)  使用<%! 方法、属性%>

2.二、3个指令

JSP指令(directive)是为JSP引擎而设计的,它们并不直接产生任何可见输出,而只是告诉引擎如何处理JSP页面中的其他部分。指令用来申明JSP页面的一些属性,好比编码方式,文档类型。咱们在servlet中也会申明咱们使用的编码方式和响应的文档类型的,而JSP就是用指令来申明。

JSP指令格式:<%@ directive {attribute=value}* %>(<%@ 指令名称 属性1=“属性值1” 属性2=“属性值2”。。。%>)
分析:
directive:指令名称,例如page指令
attribute=value:紧跟指令名称后面的就是各类属性,以键值对的形式书写
*:表明后面能跟0个或多个属性。

2.2.一、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页面进行编码
2.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时,才包含另外一个文件的内容。
原则:能用静的就不用动的。

2.2.三、taglib指令

JSP支持标签技术,后面会讲到标签的用法,jstl标签库的使用等 做用:用来指明JSP页面内使用的JSP标签库,taglib指令有两个属性,uri为类库的地址,prefix为标签的前缀 <%@ taglib uri="java.sun.com/jsp/jstl/co…" prefix="c"%>

2.三、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.3.一、<jsp:include />行为(动态包含)
<jsp:include page="/include/header.jsp"></jsp:include>
复制代码

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

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

2.3.二、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">&emsp;&emsp;做用:在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:对象的属性名

2.三、<jsp:forward />行为

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

格式:

<jsp:forward page="someServlet">
&emsp;&emsp;<jsp:param name="param1" value="value1"/>
&emsp;&emsp;<jsp:param name="param2" value="value2"/>
</jsp:forward>
复制代码

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

2.四、9个内置对象

咱们知道JSP中的内容就只有两种,模版数据和元素,元素就包括了指令,脚本,标签(行为),脚本会慢慢被标签所有代替,也就是说JSP中基本上不会嵌入Java代码,可是咱们也知道JSP会转换为servlet,在Servlet中,输出数据时,都须要经过response.getWrite();可是在JSP中,直接使用out对象进行输出,为何呢?这就是由于out为JSP的一个隐藏对象,JSP中内置了9个隐藏对象,使得JSP比Servlet使用起来更简单,更方便。

2.4.一、九大内置对象概述

分析:
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对象。

2.4.二、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)一次得到指定名称内容
pageContext中最厉害的方法是:
findAttribute(String name);  //自动从page request session application依次查找,找到了就取值,结束查找。

6)提供了的简易方法
pageContext.forward("2.jsp");
pageContext.include("2.jsp");

2.4.三、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,过程自行分析。

2.4.四、config对象

类型:ServletConfig
可以获取servlet的初始化参数,获取servletContext对象,获取servletName。

2.4.五、exception异常对象

包含了异常的信息
使用它,必须结合page指令中的isErrorPage属性和errorPage属性。
以下例子,exception.jsp抛异常的一个NullPointException,而且跳转到error.jsp错误显示页面,其中errorPage属性的意思是若是发生未捕捉到的异常,将会跳转到error.jsp页面

exception.jsp

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

error.jsp

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

    

2.4.六、总结:九大内置对象和servlet中对象的关系  

page就是jsp转换为servlet对象自己,也就是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个内置对象

2.五、4大做用域

这四大做用域,其实就是其九大内置对象中的四个,为何说他们也是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等共享账号。

相关文章
相关标签/搜索