本篇依然为基础篇,主要讲解Jsp的9个内置对象。Jsp的内置对象为Servlet API的类或接口的实例化,它们的实例化过程由Jsp标准自动进行,意即:我们可以直接使用这些对象,而不需要声明它,这些内置对象分别为:application, config,response,request,exception,out,page,pagecontext,session;接下来,我就分别对几个主要对象的特性进行详细阐述:
1.application对象
它是javax.servlet.ServletContext类的对象,一般来说,它就代表整个WEB应用,对于它的属性值,在整个WEB应用(JSP或Servlet类)中都是公用的,最常用的方法是:setAttribute(name,value)和getAttribute(name);为此来做个测试:在同一个WEB工程中编写两个JSP文件,index.jsp如下:
<%@ page language="java" import="java.util.*" %> <%@ page pageEncoding="ISO-8859-1"%> <%! private int count = 0; public Integer getCount() { count ++ ; return count; } %> <% application.setAttribute("count",getCount().toString()); out.println("the value of count is :"+count); %>
test_application.jsp内容如下:
<%@page language="java" contentType="text/html; charset=ISO-8859-1" %>
测试application <%out.println("get the value of count,the count is a field of application,value: "); %> <%=application.getAttribute("count")%>
我们先启动服务器,访问http://localhost:8080/index.jsp,其结果如下:the value of count is :2,
每刷新一次页面,count的值就增1,此时我们再访问http://localhost:8080/test_application.jsp,其结果显示如下:
get the value of count,the count is a field of application,value: 2
没有进行任何参数传递,也没有直接获取javabean的属性,而是通过application对象,先设置属性(application.setAttribute("count",getCount().toString());),再从另一个页面中获得属性(application.getAttribute("count"))就能做到属性的全局共享,这便是使用application对象的作用。
application对象的另外一个强大作用就是能用来解析WEB.XML的配置信息,以下就是一个完整的连接和查询mysql数据库的例子,而访问数据库的url,密码,用户名等信息,均存储在web.xml中,在JSP文件中用application对象读取参数属性。
<%@ page language="java" import="java.util.*" %> <%@ page pageEncoding="ISO-8859-1"%> <%@page import="java.sql.SQLException"%> <%@page import="java.net.ConnectException"%> <%@page import="java.sql.DriverManager"%> <%@page import="java.sql.*" %> <% Connection conn=null; Statement state=null; ResultSet rs=null; String driver=application.getInitParameter("driver"); String url=application.getInitParameter("url"); String user=application.getInitParameter("user"); String password=application.getInitParameter("password"); try{ Class.forName(driver); conn=DriverManager.getConnection(url,user,password); }catch(SQLException e) { out.println("failed to connect the db"); } catch(ClassNotFoundException e){ out.println("can't find the driver class"); } try{ state=conn.createStatement(); String queryAll="select*from user_information"; rs=state.executeQuery(queryAll); while(rs.next()) { int userid=rs.getInt(1); String username=rs.getString(2); String userpassword=rs.getString(3); out.println("user id is: "+userid); out.println("user name is: "+username); out.println("user password is: "+userpassword); } }catch(SQLException e){ out.println("research fialed!"); } try{ if(rs != null){ rs.close(); } if(state != null){ state.close(); } if(conn != null){ conn.close(); } }catch(Exception e){ out.println("error when close db"); } %>
web.xml的完整信息如下:
index.jsp driver com.mysql.jdbc.Driver user root password 900622 url jdbc:mysql://localhost:3306/test_db
页面执行结果为:user id is: 1 user name is: James user password is: 123456
2. pageContext对象
pageContext对象可以用来访问和设置各种范围的属性(包括page,request,session和application),事实上,如果给各种范围的属性设置成相同的属性名,那么各属性是互不影响的,这就相当于C++中同名变量在不同的生存范围内互不影响(如同名的全局变量和局部变量),而pageContext通常用来获得和设置各种范围的属性值,看下面实例:
<%
//使用不同内置对象设置不同范围属性
pageContext.setAttribute("name","page_James"); //设置page范围属性
request.setAttribute("name","request_James"); //设置request范围属性
session.setAttribute("name","session_James"); //设置session范围属性
application.setAttribute("name","application_James"); //设置application范围属性
//使用pageContext设置不同范围属性
pageContext.setAttribute("age","30"); //不指定属性的范围时默认为page范围
pageContext.setAttribute("age","40",pageContext.REQUEST_SCOPE);
pageContext.setAttribute("age","50",pageContext.SESSION_SCOPE);
pageContext.setAttribute("age","60",pageContext.APPLICATION_SCOPE);
%>
<%="page范围的name值:"+(String)pageContext.getAttribute("name",pageContext.PAGE_SCOPE)%>
<%="page范围的age值: "+(String)pageContext.getAttribute("age",pageContext.PAGE_SCOPE) %>
<%="request范围name值: "+(String)pageContext.getAttribute("name",pageContext.REQUEST_SCOPE)%>
<%="request范围age值:"+(String)pageContext.getAttribute("age",pageContext.REQUEST_SCOPE) %>
<%="session范围name值: "+(String)pageContext.getAttribute("name",pageContext.SESSION_SCOPE) %>
<%="session范围age值:"+(String)pageContext.getAttribute("age",pageContext.SESSION_SCOPE) %>
<%="application范围name值:"+(String)pageContext.getAttribute("name",pageContext.APPLICATION_SCOPE) %>
<%="application范围name值:"+(String)pageContext.getAttribute("age",pageContext.APPLICATION_SCOPE) %>
page范围的name值:page_James
page范围的age值: 30
request范围name值: request_James
request范围age值:40
session范围name值: session_James
session范围age值:50
application范围name值:application_James
application范围name值:60
3. request对象
request对象是个很重要的对象,它被用于封装一次的用户请求参数,请求参数可以是表单值,可以是以request对象设置的属性值,也可以是url地址中所附带的少量字符串参数。 我们来看下面实例:
index.jsp的body部分
<%
ArrayList list = new ArrayList() ;
list.add("this");
list.add(" is");
list.add(" the");
list.add(" list");
for(int i = 0 ;i
request.jsp的body部分:
<%="user name is: "+request.getParameter("username") %>
<%="user password is: "+request.getParameter("userpwd") %>
<%
ArrayList list = (ArrayList)request.getAttribute("list");
for(int i = 0 ;i
user name is :james
user password is: admin
this is the list
而事实上,请求提交后,页面出错,显示一堆错误信息。
在这里要注意,提交请求和页面跳转是两个不同的概念,范围属性和请求参数也是两个不同的概念,提交请求后,请求参数会被servlet封装到request对象中,这一点毋庸置疑,但不会将属性也封装到servlet中,只有当发生< jsp:forward>跳转时,才会将属性提交到要转发的页面,如果我们将index.jsp改为如下:
<%
ArrayList list = new ArrayList() ;
list.add("this");
list.add(" is");
list.add(" the");
list.add(" list");
for(int i = 0 ;i
<%
ArrayList list2 = new ArrayList();
list2 = (ArrayList)pageContext.getAttribute("list",pageContext.REQUEST_SCOPE);
for(int i = 0 ;i
// !!!!!注意多加了这一句!!!
那么request.jsp的显示内容将又会为:
user name is:null
user password is:null
this is the list
index.jsp中未提交请求前就发生跳转,所以name和password值没有设置,而forward跳转会传递属性,所以会在跳转页面request.jsp中正常接收。
4. response和out对象
response对象和out对象都是用来向页面输出信息的流对象,其不同之处在于:out对象只能输出字符流(只能输出文字),却不能输出字节流(如位图),而response对象却可以做到这一点;在大多数情况下,out完全可以替代response的作用,而有时候response的功能却不容忽视,甚至起关键作用:如可以创建cookie,可以设置定时刷新的页面,可以不使用forward跳转来实现不带属性的跳转。 以下来看一下response创建cookie的示例:
在index.jsp中代码如下:
<%
String userName = request.getParameter("username");
String userPassword = request.getParameter("userpwd");
Cookie nameCookie = new Cookie("username",userName); //创建cookie
nameCookie.setMaxAge(24*3600);//设置生存时间为24小时
Cookie pwdCookie = new Cookie("userpwd",userPassword);
pwdCookie.setMaxAge(24*3600);
response.addCookie(nameCookie); //添加cookie
response.addCookie(pwdCookie);
%>
这样,凡是客户端输出用户名和密码的话,其便会存储在客户端的硬盘中,24小时后就自动清除,如此可以提供更友好的服务。
我们继续创建一个验证页: prove.jsp,其代码如下:
<%
Cookie cookies[] = request.getCookies();
for(Cookie c:cookies)
{
if(c.getName().equals("username") || c.getName().equals("userpwd"))
{
out.println(c.getName()+":"+c.getValue());
}
}
%>
此时页面显示结果为:
username:James userpwd:admin
事实上,在24小时内任意时间段输入http://jxpc-001:8080/ExtJs/request.jsp,那么网页最终都会显示为这样的结果。
5. session对象
对于session对象,我不打算多述,因为其作用基本与request相同,只是生命周期比request要长,从浏览器打开到关闭浏览器,如同一个单网页下设置了一个session范围属性和request范围属性,当页面关闭时,session范围属性我们可以不用forward跳转而在任何地方获得,其都不会丢失,而request范围属性,在关闭网页后,任何一个非forward跳转页中均获取不到,因为它的生命周期只在一次forward之间,断开这次forward,其属性会丢失掉。