JSP页面中四种变量的范围
我们在编写JSP网站程序的时候,可以在WEB应用程序的不同范围中保存变量,
JSP中变量的范围分为四种:page,request,session和application。
这四个范围依序从前到后,范围越来越大。
一、page范围
在page范围内定义的变量,只能在同一个页面内部访问。在JSP页面中,演示如下代码:
<%
pageContext.setAttribute("name","Johnson");
out.println(pageContext.getAttribute("name"));
%>
上面的代码中,在page范围内,定义了一个名称为name的属性,其值为Johnson,
第二行代码将该属性的值打印出来。注意:这样定义的这个name属性,在这个页面之外
的部分是访问不到的。考虑到page范围只是在一个页面中共享数据,其范围是非常小的。
需要注意的是,在page范围,以及其他三种范围中,存储的值都只能是引用类型的值。
下面的例子演示怎么将一个整数存入page范围之内。
<%
pageContext.setAttribute("amount",new Integer(100));
out.println(pageContext.getAttribute("amount"));
%>
二、request范围
在Java的WEB应用程序中,可以将一个变量值存入request范围中,然后可以将request
对象传递给下一个页面,那么下一个页面中,可以获得从上一个页面传送过来的request
对象中保存的属性值。
这里,我们修改上一章中的登陆案例,来演示request范围属性的使用。
我们希望用户登录成果以后,能够在登陆成功页面,即welcome.jsp页面中,显示
该登陆用户的用户名,那么可以修改前面登陆案例中的验证登陆用户的servlet如下:
public void doPost(HttpServletRequest request, HttpServletResponse response) throws
ServletException, IOException {
String name=request.getParameter("name");
String pwd=request.getParameter("pwd");
User user=new User();
user.setName(name);
user.setPwd(pwd);
String dbName=request.getRealPath("goods.mdb");
//转向器
String url="";
try {
db d1 = new db(dbName);
if(d1.isUserValid(user))
{
request.setAttribute("name",name);
url="welcome.jsp";
}
else
{
url="failed.jsp";
}
RequestDispatcher rd=request.getRequestDispatcher(url);
//把用户名存入request scope中
//设置属性
rd.forward(request,response);
}
catch (Exception ex) {
ex.printStackTrace();
}
}
在上面的代码中,用户登陆成功以后,把用户名存入request范围,
request.setAttribute(“name”,name);
然后获取将要转向的页面的转向器,
url=”welcome.jsp”;
RequestDispatcher rd=request.getRequestDispatcher(request,response);
这样,本servlet中的request对象就会被传递到下一个页面,在下一个页面,
即welcome.jsp中,就可以通过
request.getAttribute(“name”);
的方式获取到name属性,即登陆用户的用户名。
三、session范围
当用户通过浏览器和服务器连通以后,服务器为该用户在服务器上分配一个供该用户专
用的内存空间,这个内存空间中保存着用户访问该网站服务器期间的一些信息,这些信
息称为会话信息,也就是Session信息。服务器为每个访问该服务器的连接分配一个唯
一的编号,该编号用以唯一确认该用户。这个编号就是我们通常说的SessionID。
有了这个会话信息空间以后,用户就可以把和本次访问相关的一些信息保存在这个会话
信息空间中。会话信息空间中保存的信息只和本次连接有关。
需要注意的是,通过同一台主机打开多个浏览器访问服务器的时候,每一个浏览器连
接对应一个会话。要设置一个session范围的变量,可以使用下面的方法。
<%
session.setAttribute("name",”Johnson”);
out.println(session.getAttribute("name"));
%>
四、application范围
一台WEB服务器启动以后,在服务器上有一个由所有连接共享的内存空间,
这就是application范围空间。在application范围中定义的变量,可以由所有连接共享。
一个连接对application范围中变量的修改,会体现到另外一个连接中。
要设置一个application范围的变量,可以使用如下方法:
<%
application.setAttribute("name",”Johnson”);
out.println(application.getAttribute("name"));
%>
JSP标准标签库(JSP Standard Tag Library,JSTL)是实现Web应用程序中常见的通用功能的定制标签库集,程序员使用JSTL标签来避免在JSP页面中使用脚本。
JSTL概述
(1)、JSTL英文全称是“JSP Standard Tag Library”,即JSP标准标签库之意。
(2)、JSTL是由JCP(JavaCommnunity Process)指定的标准规范,是一组形如html的标签。
(3)JSTL提供了诸如循环、条件、数据库访问、XML处理、国际化等开发上的工具和流程。
(4)、目前最新的版本为1.2,是一个正在不断开发和完善的开发源代码的JSP标签库,它支持多种标签。
JSTL分成了5大类
功能范围
URL
前缀
核心标签库(Core)
http://Java.sun.com/jsp/jstl/core
c
国际化/格式化标签库(il8n)
http://java.sun.com/jsp/jstl/fmt
fmt
数据库标签库吧(SQL)
http://java.sun.com/jsp/jstl/sql
sql
XML标签库(XML)
http://java.sun.com/jsp/jstl/xml
X
Functions标签库(Functions)
http://java.sun.com/jsp/jstl/functions
fn
(5)、在学习JSTL标签库之前,需要先下载JSTL所需要的JAR包。有两种获取方法
1)通过官方网站(http://www.apache.org/dist/jakarta/taglibs/standard)下载,获取API里面的jstl.jar、standard.jar
2)使用MyEclipse自带的驱动包。使用MyEclipse创建Web工程时,选择“JSTL Support”选项。
Core标签库
(1)、核心标签库主要包括通用标签、条件标签、迭代标签和与URL相关的标签
(2)、在使用Core标签库的JSP文件的开始部分,添加代码:%@tagliburi=”http://java.sun.com/jsp/jstl/core” prefix=”c”%
1、通用标签
<c:out>标签
用于将表达式的结果输出到当前的JSPWriter对象中。其功能类似于JSP的表达式<%= %>或者EL表达式${}。
request.setAttribute("user","lisi");
%>
<c:out value="helloworld"/><br>
hello world ${"hello world"}<br>
<c:out value="${userName}"default="xxxx"></c:out><br>
<c:out value="<h1>你好</h1>" escapeXml="false"></c:out><br>
<c:out value="${userName}">文本内容</c:out>
<c:set>标签
<c:set>标签用于设置JSP页面的各种域范围中的变量,或者设置java.util.Map对象或JavaBean对象的属性
将value的值保存到名为name的变量中,name变量保存到选定的作用范围中
<c:set value="孙悟空" var="user"scope="request"/>
${user}
<c:set var="user"scope="request">猪八戒</c:set>
${user}
<%
Map map=new HashMap();
request.setAttribute("map",map);
%>
<c:set value="唐曾" property="aaa"target="${map}"></c:set>
${map.aaa}
<%
Personp=new Person();
request.setAttribute("person",p);
%>
<c:set value="Jerry" property="name"target="${person}"></c:set>
${person.name}
<c:remove>标签
<c:remove>标签用于移除JSP页面中指定域范围中的变量
<c:remove var="person"scope="request"/>
<c:out value="${person.name}" default="没有该人"></c:out>
<c:catch>标签
<c:catch>标签用于捕获嵌套在标签体内的内容抛出的异常对象,并将异常信息保存到变量中
<c:catch var="myex">
<%
int i=10/0;
%>
</c:catch>
异常:<c:out value="${myex}"></c:out><br>
异常原因:<c:out value="${myex.cause}"></c:out><br>
异常消息:<c:out value="${myex.message}"></c:out><br>
异常堆栈轨迹:<c:out value="stackTrace"></c:out><br>
条件标签
<c:if>标签
<c:if>标签用来做条件判断,功能类似于JSP中的<%if(boolean){}%>
<c:if test="${userName==null}" var="xx"scope="request">
改用户没有登录
</c:if>
${xx}
<c:choose>、<c:when>、<c:otherwise>标签
<c:choose>标签用于提供条件选择的上下文,它必须与<c:when>和<c:otherwise>标签一起使用
<c:when>作为<c:choose>的子标签,<c:when>有一个test属性,该属性的值为布尔型,如果test的值为true,则执行<c:when>标签体的内容。
<c:otherwise>标签没有属性,它必须作为<c:choose>标签的最后分支出现。
<%
request.setAttribute("age",18);
%>
<c:choose>
<c:when test="${age>70}">老年人</c:when>
<c:when test="${age<=70&&age>35}">中年人</c:when>
<c:when test="${age<=35&&age>18}">青年</c:when>
<c:when test="${age<=18&&age>0}">少年或儿童</c:when>
<c:otherwise>输入错误</c:otherwise>
</c:choose>
迭代标签
<c:forEach>标签
<c:forEach>标签用于对包含了多个对象的集合进行迭代,重复执行它的标签体,或者重复迭代固定的次数。
注意:
items属性并不是一定要有的属性,但是当没有使用items属性时,就一定要使用begin和end属性。
varStatus:用来保存现有遍历计数相关的信息。例如,如果varStatus=“i”,那么会将相应信息放到名称为i的变量中,i变量有四个属性值,index、count、first和last。
<c:forEach begin="1"end="10" step="1">
helloworld
</c:forEach>
<br>
<%
List list=new ArrayList();
list.add("aa");
list.add("bb");
list.add("cc");
list.add("dd");
list.add("ee");
list.add("ff");
list.add("gg");
list.add("hh");
list.add("ii");
list.add("jj");
request.setAttribute("list",list);
%>
<table border="1"width=50%>
<c:forEach items="${list}"var="str" begin="0"end="7" step="1"varStatus="status">
<tr class="${(status.count%2!=0)?'even':'odd'}">
<td>${str}</td>
<td>${status.index}</td>
<td>${status.first}</td>
<td>${status.last}</td>
<td>${status.count}</td>
</tr>
</c:forEach>
</table>
<c:forTokens>标签
<c:forTokens>标签用来浏览一字符串中所有的成员,其成员是由定义符号(Delimiters)所分隔的。
<c:forTokens items="this:is:an:example" delims=":" var="token">
${token}
</c:forTokens>
URL相关的标签
<c:url>标签
<%
session.setAttribute("xxx","xxx");
%>
<c:url value="/index.jsp" var="strUrl" scope="request">
<c:param name="userName"value="王五"></c:param>
</c:url>
${strUrl}
<a href="${strUrl}">首页</a> <br>
<a href="<c:url value='/index.jsp'/>">首页</a>
<c:redirect>标签
<c:redirect>标签用于向用户的浏览器发送HTTP重定向响应,它是JSTL中与javax.servlet.http.HttpServletResponse的sendRedirect()方法功能相当的标记。
<c:redirect url="/MyJsp.jsp"></c:redirect>
<c:param>标签
<c:param>标签的作用是为一个URL添加请求参数,在前面的<c:url>、<c:redirect>和<c:import>标签中都已经见过<c:param>的用法。
<c:import url="/MyJsp.jsp"var="secondPage"></c:import>
使用标签控制页面逻辑案例:
1、 防止盗链<c:referer site=”” page=””>
Jsp页面的内容:<c:referer site=”http://localhost”page=”index.jsp”>
Tld文件的描述:<tag>
<name>referer</name>
<tag-class>com.hbsi.web.tag.RefererTag</tag-class>
<body-content>empty</body-content>
<attribute>
<name>site</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
<attribute>
<name>page</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
标签处理器:public class RefererTag extends SimpleTagSupport {
private String site;
private String page;
public void setSite(String site) {
this.site = site;
}
public void setPage(String page) {
this.page = page;
}
@Override
public void doTag() throws JspException, IOException {
// 获取request对象
JspContext jc = this.getJspContext();
PageContext pc = (PageContext) jc;
HttpServletRequest request = (HttpServletRequest)pc.getRequest();
HttpServletResponse response = (HttpServletResponse) pc.getResponse();
String referer = request.getHeader("referer");
if (referer == null || !referer.startsWith("http://localhost")) {
// 跳转到首页
//获取地址的完整路径
if (page.startsWith(request.getContextPath())) {
response.sendRedirect(page);
} else if (page.startsWith("/")) {//获取加/的路径
response.sendRedirect(request.getContextPath() + page);
} else {
//最不完整的路径
response.sendRedirect(request.getContextPath() + "/" + page);
}
// 不运行以下的代码
throw new SkipPageException();
}
}
}
2、条件选择<c:if test=””>
Jsp的页面:<%session.setAttribute("user","蛋蛋");%>
<c:if test="${user!=null}">
aaa
</c:if>
Tld文件的描述:<tag>
<name>if</name>
<tag-class>com.hbsi.web.tag.IfTag</tag-class>
<body-content>scriptless</body-content>
<attribute>
<name>test</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
标签处理器:public class IfTag extends SimpleTagSupport {
private boolean test;
public void setTest(boolean test) {
this.test = test;
}
@Override
public void doTag() throws JspException,IOException {
if(test){
JspFragmentjf=this.getJspBody();
jf.invoke(null);
}
}
}
3、迭代标签<c:foreach items=”” var=””>
Jsp的页面:< c:foreach items="${list}" var="str">
${str}
</c:foreach>
Tld文件的描述:<tag>
<name>foreach</name>
<tag-class>com.hbsi.web.tag.ForEachTag</tag-class>
<body-content>scriptless</body-content>
<attribute>
<name>items</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
<attribute>
<name>var</name>
<required>true</required>
<rtexprvalue>false</rtexprvalue>
</attribute>
</tag>
标签处理器:public class ForEachTag extends SimpleTagSupport {
private Object items;
private String var;
public void setItems(Object items) {
this.items = items;
}
public void setVar(String var) {
this.var = var;
}
@Override
public void doTag() throws JspException, IOException {
//得到集合对象
Collection collection=null;
if(items instanceof Map){
Map map=(Map)items;
//两列集合转换成单列;映射
collection=map.entrySet();
}else if(items instanceof Collection){
collection=(Collection)items;//集合
}else if(items.getClass().isArray()){
collection=new ArrayList();
int length=Array.getLength(items);
for(int i=0;i<length;i++){
collection.add(Array.get(items, i));
}
}
Iterator it=collection.iterator();
while(it.hasNext()){
Object obj=it.next();//一个元素
this.getJspContext().setAttribute(var,obj);
this.getJspBody().invoke(null);
}
}
}