JSP生命周期
由于在Servlet中编写HTML很麻烦,可以使用JSP来编写HTML。JSP与Servlet是一体的两面。因为JSP最后还是会被容器转译为Servlet源代码、自动编译为.class文件、载入.class文件,然后生成Servlet对象。
在编写Servlet时,可以重新定义init()方法作Servlet的初始化,重新定义destroy()进行Servlet销毁前的收尾工作。JSP在转译为Servlet并载入容器生成对象之后,会调用_jspInit()方法进行初始化工作,而销毁前则是调用_jspDestroy()方法进行善后工作。在Servlet中,每个请求到来时,容器会调用service()方法,而在JSP转译为Servlet后,请求的到来则是调用_jspService()方法。
Servlet到JSP的转换
指示元素
<%@指示类型 ... %>
JSP指示元素的主要目的,在于指示容器将JSP转译为Servlet源代码时一些必须遵守的信息。
常用的指示类型有以下三种:
- page 告知容器如何转译目前的JSP网页
- include 告知容器将别的JSP页面包括进来进行转译
- taglib 告知容器如何转译这个页面的标签库
<%@page import="java.util.Date" %>
<%@page contentType="text/html" pageEncoding="UTF-8" %>
声明元素
<%! 类成员声明或方法声明 %>
声明元素是指它用来声明类成员与方法。
<%!
String name = "caterpillar";
String password = "123456";
boolean checkUser(String name, String password) {
return this.name.equals(name) &&
this.password.equals(password);
}
%>
Scriptlet元素
<% JAVA语句 %>
Scriptlet元素是指可以用它来编写JAVA语句。
<%
String name = request.getParameter("name");
String password = request.getParameter("password");
if(checkUser(name, password)) {
%>
<h1>登录成功</h1>
<%
}
else {
%>
<h1>登录失败</h1>
<%
}
%>
表达式元素
<%= JAVA表达式 %>
表达式元素是指可以在其中编写JAVA表达式,表达式的运算结果将直接输出为网页的一部分。
<%= new Date() %>
//不用加分号
注释元素
// 单行注释
/* */ 多行注释
<!-- --> HTML使用的注释
隐式对象
JSP隐式对象是JSP容器为每个页面提供的Java对象,开发者可以直接使用它们而不用显式声明。JSP隐式对象也被称为预定义变量。
JSP所支持的九大隐式对象:
对象 | 描述 |
---|---|
request | HttpServletRequest类的实例 |
response | HttpServletResponse类的实例 |
out JspWriter | 类的实例,用于把结果输出至网页上 |
session | HttpSession类的实例 |
application | ServletContext类的实例,与应用上下文有关 |
config | ServletConfig类的实例 |
pageContext | PageContext类的实例,提供对JSP页面所有对象以及命名空间的访问 |
page | 类似于Java类中的this关键字 |
Exception | Exception类的对象,代表发生错误的JSP页面中对应的异常对象 |
request 对象:
request对象是javax.servlet.http.HttpServletRequest类的实例。每当客户端请求一个JSP页面时,JSP引擎就会制造一个新的request对象来代表这个请求。
request对象提供了一系列方法来获取HTTP头信息,cookies,HTTP方法等等。
response 对象:
response对象是javax.servlet.http.HttpServletResponse类的实例。当服务器创建request对象时会同时创建用于响应这个客户端的response对象。
response对象也定义了处理HTTP头模块的接口。通过这个对象,开发者们可以添加新的cookies,时间戳,HTTP状态码等等。
out 对象:
out对象是 javax.servlet.jsp.JspWriter 类的实例,用来在response对象中写入内容。
最初的JspWriter类对象根据页面是否有缓存来进行不同的实例化操作。可以在page指令中使用buffered=’false’属性来轻松关闭缓存。
JspWriter类包含了大部分java.io.PrintWriter类中的方法。不过,JspWriter新增了一些专为处理缓存而设计的方法。还有就是,JspWriter类会抛出IOExceptions异常,而PrintWriter不会。
下表列出了我们将会用来输出boolean,char,int,double,String,object等类型数据的重要方法:
方法 | 描述 |
---|---|
out.print(dataType dt) | 输出Type类型的值 |
out.println(dataType dt) | 输出Type类型的值然后换行 |
out.flush() | 刷新输出流 |
session 对象:
session对象是 javax.servlet.http.HttpSession 类的实例。和Java Servlets中的session对象有一样的行为。
session对象用来跟踪在各个客户端请求间的会话。
application 对象:
application对象直接包装了servlet的ServletContext类的对象,是javax.servlet.ServletContext 类的实例。
这个对象在JSP页面的整个生命周期中都代表着这个JSP页面。这个对象在JSP页面初始化时被创建,随着jspDestroy()方法的调用而被移除。
通过向application中添加属性,则所有组成您web应用的JSP文件都能访问到这些属性。
config 对象:
config对象是javax.servlet.ServletConfig类的实例,直接包装了servlet的ServletConfig类的对象。
这个对象允许开发者访问Servlet或者JSP引擎的初始化参数,比如文件路径等。
以下是config对象的使用方法,不是很重要,所以不常用:
config.getServletName();
它返回包含在< servlet-name>元素中的servlet名字,注意,< servlet-name>元素在 WEB-INF\web.xml 文件中定义。
pageContext 对象:
pageContext对象是javax.servlet.jsp.PageContext 类的实例,用来代表整个JSP页面。
这个对象主要用来访问页面信息,同时过滤掉大部分实现细节。
这个对象存储了request对象和response对象的引用。application对象,config对象,session对象,out对象可以通过访问这个对象的属性来导出。
pageContext对象也包含了传给JSP页面的指令信息,包括缓存信息,ErrorPage URL,页面scope等。
PageContext类定义了一些字段,包括PAGE_SCOPE,REQUEST_SCOPE,SESSION_SCOPE, APPLICATION_SCOPE。它也提供了40余种方法,有一半继承自javax.servlet.jsp.JspContext 类。
其中一个重要的方法就是removeArribute(),它可接受一个或两个参数。比如,pageContext.removeArribute(“attrName”)移除四个scope中相关属性,但是下面这种方法只移除特定scope中的相关属性:
pageContext.removeAttribute(“attrName”, PAGE_SCOPE);
page 对象:
这个对象就是页面实例的引用。它可以被看做是整个JSP页面的代表。
page 对象就是this对象的同义词。
exception 对象:
exception 对象包装了从先前页面中抛出的异常信息。它通常被用来产生对出错条件的适当响应。
表达式语言(EL)
JSP中若有Scriptlet编写JAVA代码以进行属性、请求参数、标头与Cookie等信息的取得,或一些简单的运算或判断,可以试着使用EL来取代,以减少JSP页面上Scriptlet的使用。
例如:
<%
String a = request.getParameter("a");
String b = request.getParameter("b");
out.println("a + b = " + (Interger.parseInt(a) + (Interger.parseInt(b)));
%>
//使用EL:
...
<body>
${param.a} + ${param.b} = ${param.a + param.b}
</body>
...
使用EL可以将输入的请求参数自动转换为基本类型并进行运算,且在结果中还增加了显示操作数的功能;并且对于null值直接以空字符串加以显示,而不是直接显示null值,在进行运算时,也不会因此发生错误而抛出异常。
EL的点运算符还可以连续存取对象:
<%= ((HttpServletRequest) pageContext.getRequest()).getMethod() %>
//使用EL:
${pageContext.request.method}
使用EL取得属性
例1:若网页的某处在请求范围中设置了数组作为属性
<%
String[] names = {"a", "b", "c"};
request.setAttribute("array", names);
%>
//取出并访问数组元素
名称一:${array[0]}<br>
名称二:${array[1]}<br>
名称三:${array[2]}<br>
例2:若想取得Map对象中的值
<%
Map(String, String) map = new HashMap<String, String>();
map.put("user", "caterpillar");
map.put("role", "admin");
request.setAttribute("login", map);
%>
//使用点运算符或[]运算符
User:${login.user}<br>
Role:${login.role}<br>
//更推荐[]
User:${login["user"]}<br>
Role:${login["role"]}<br>
EL隐式对象
JSP隐式对象类中只有一个EL隐式对象,这就是pageContext隐式对象。这与同名的JSP隐式对象实际上就是同一个对象。余下的EL隐式对象都是Java映射(map),他们只是提供了更容易的途径来访问pageContext隐式对象的某些性质。
共有4个作用域隐式对象,分别为pageScope, requestScope, sessionScope和applicationScope。这些隐式对象都是映射,利用他们可以很容易的访问作用域属性。例如:附加到request作用域的username属性可以通过EL表达式${ requestScope.username}来直接访问。
有2个参数访问隐式对象,可以用来访问HTTP请求参数(表单提交参数),即param和paramValues. param 是一个用于访问单值参数的映射, paramValues则可用于访问可能包含多个值的参数。
有3个首部访问隐式对象,可以用于访问HTTP首部,分别是header, headerValues和cookie。如果想以原始的方式访问HTTP首部或cookie,这些映射就很有用。
另外还有一个初始化参数访问隐式对象:initParm。这个映射可以用于访问初始化参数的值,初始化参数的值一般都在web.xml中设置。
类别 | 标识符 | 描述 |
---|---|---|
JSP | pageContext | PageContext 实例对应于当前页面的处理 |
作用域 | pageScope | 与页面作用域属性的名称和值相关联的 Map 类 |
requestScope | 与请求作用域属性的名称和值相关联的 Map 类 | |
sessionScope | 与会话作用域属性的名称和值相关联的 Map 类 | |
applicationScope | 与应用程序作用域属性的名称和值相关联的 Map 类 | |
请求参数 | param | 按名称存储请求参数的主要值的 Map 类 |
paramValues | 将请求参数的所有值作为 String 数组存储的 Map 类 | |
请求头 | header | 按名称存储请求头主要值的 Map 类 |
headerValues | 将请求头的所有值作为 String 数组存储的 Map 类 | |
Cookie | cookie | 按名称存储请求附带的 cookie 的 Map 类 |
初始化参数 | initParam | 按名称存储 Web 应用程序上下文初始化参数的 Map 类 |
EL运算符
使用EL可以直接进行一些算术运算、逻辑运算与关系运算。
比如:
${1} --- 1
${1+2} --- 3
${3/4} --- 0.75
${3/0} --- Infinity
${(1==2)?3:4} --- 4
${true and false} --- false
${not true} --- false
${1 < 2} --- true