原来准备简单点,用一篇文章介绍完JSP的,没想到细节很多。这是第三篇也是最后一篇,这三篇文章介绍了基本的JSP知识,很多名词概念和用法,一定要在实践项目中使用之后才会有更加亲切的体会。第三篇接着上篇介绍的JSP内置对象。
上篇我们说过,在我们的响应用户请求的方法_jspservice()方法的头部,定义和初始化了一些局部变量。他们其实就是我们现在说的jsp内置对象,通过这些对象所封装的方法,我们可以完成一些关于数据共享,数据传输,浏览器响应等操作。
第一个内置对象是,application。熟悉.net的朋友可能知道,在.net中,可以使用session,cookies来实现多个不同页面之间数据的共享,在我们JSP 中上述有着更加完善的共享数据机制,上述的session和cookies只是其中的一部分。所有被application设置访问范围的数据在整个web应用中都是可见的,无论你是jsp页面还是servlet页面,只要是位于当前的web应用中的页面都是可以访问该数据的。这叫全局可见性,是所有共享数据中范围最大的。具体的使用方法如下:
1
index.jsp
<%application.setAttribute("name","张三");%>
2
convert.jsp
<%=application.getAttribute("name")%>
访问过index页面之后访问convert页面,可以看到我们在index页面设置的全局性变量,在另外的页面是可以访问的到的。这种是将数据设置在整个web应用下的,所以在可以放在更低范围内的情况下(后文会介绍在更小的范围设置共享数据),尽量还是不要设置在全局下,因为这样很有可能在不经意之间被修改了值产生数据不一致性。
application对象的第二个作用是可以读取web.xml中的配置信息,这是一种可能会经常使用到的操作。因为我们的数据库用户名和密码等参数的配置一般都是配置在web.xml中的,当我们在程序中需要用到的时候,就可以考虑这种操作来读取配置信息。
/首先在web.xml文件中配置一下信息/
1
index.jsp
<%=application.getInitParameter("a")%>
通过上述操作,我们成功的访问了配置信息内容,web.xml中< context-param>标签中的就是类似于键值对的作用,子标签配置的是name和value和我们的param动作指令的传参是类似的,之后便可以同过方法getInitParameter("name")来获取对应的value的值。
第二个内置对象,pageContext。这是一个非常大佬的对象,他可以获取其他的内置对象。
ServletReques getRequest(); //获取Request对象
ServletResponse getResponse(); //获取Response对象
SerletConfig getServletConfig();//获取config对象
ServletContext getServletContext();//获取application对象
HttpSession getSession(); //获取session对象
以上的有些对象我们说了,有些对象会在后文中说明,有些暂时不会介绍,原因是他们不是很常用。
pageContext对象的主要作用还是对共享数据的范围设定。
pageContext.setAttribute("name","li",pageContext.PAGE_SCOPE); //设置变量位于当前page范围内可见
pageContext.setAttribute("name","li",pageContext.REQUEST_SCOPE); //设置变量位于本次请求中可见
pageContext.setAttribute("name","li",pageContext.SESSION_SCOPE); //设置变量位于当前会话范围内可见(就是浏览器关了就没了)
pageContext.setAttribute("name","li",pageContext.APPLICATION_SCOPE); //设置变量位于当前Web应用范围内可见
当然,对应的获取方法是:pageContext.getAttribute("name")。
第三个内置对象是,request对象。这个对象封装了用户的某次请求的所有内容,有浏览器自动生成的请求头,还有用户提交的请求参数。请求头对于我们来说暂时可能没怎么使用到,但是请求参数还是需要好好的研习一下的。下面提供了几个获取请求参数的方法:
String getParameter(String paramName) //获取指定name的参数值
Map getParameterMap() //获取所有请求参数名和请求参数值,以键值对的形式返回
Enumeration getParameterNames() //获取所有请求参数名构成的Enumeration集合
String[] getParameterValues(String name) //获取指定name的所有参数值,可能一个参数name对应了多个参数值
1
提交表单
1
index.jsp
<%=request.getParameter("name")%>
<%=request.getParameter("age")%>
相信大家已经知道结果了,我们知道浏览器的请求方法是有两种的,Get和Post请求,两种请求方式的优劣相信大家都知道,此处不再啰嗦。虽然是两种请求方式,但是我们在获取请求参数的数值的时候是一样的方法,不要做另外的改变。以上至演示了一种形式,其他形式获取参数的情况大家可以自行测试。
request对象还有一个操作也是经常会做的,官方说法,操作request范围的属性。其实就是将某个变量放到request请求参数中,在接受页面出可以接受该参数。
1
index.jsp
<%request.setAttribute("abc","hello");%>
//实现转发
2
convert.jsp
<%=request.getAttribute("abc")%>
我们可以简单的理解为,这个操作就是将指定的参数添加到请求本页面的这个request对象中,一旦此页面实现跳转,必然可以携带者这个参数转发到另外的页面中。
最后,我们说说这个response内置对象。刚才我们介绍的request是处理用户请求的,主要是接受用户传入的请求参数,或者解析请求头信息等。而正真决定怎么响应我们用户的是response对象。
首先我们看看第一个作用,response的getOutputStream()方法,这个方法返回响应输出字节流,一个OutputStream对象。我们在介绍流那块知识的时候说过,每个流都会绑定一个文件,这个文件可以是字节数组,或者磁盘上的实际文件,因为这些操作最终都会流到实际的文件中。(不可能是一个抽象不存在的文件,那我们的一切操作岂不是操作空气,又没有改变什么),此处可以将getOutputStream方法理解为绑定了当前请求的浏览器,也就是我们队流进行写入会写到浏览器中。看个例子:
1
index.jsp
<%
response.setHeader("content-type", "text/html;charset=UTF-8");
OutputStream outputStream = response.getOutputStream();
String str = "hello world";
byte[] buffer = str.getBytes();
outputStream.write(buffer);
%>
结果显而易见,但是不要忘记引入Java.io.* 这个包,否则会报错,不能识别OutputStream。
第二个用法,页面重定向。使用的是方法,response.sendRedirect(String path)。这个页面重定向和之前的forward动作指令是不一样的,之前的forward指令不会丢弃当前request的请求参数,而这也页面重定向会丢弃所有的请求参数,重新开一个request,并且地址栏的地址也是会随之改变的。具体代码很简单,不演示了,要不然就写不完了。
最后一个response对象的操作是,增加cookies,我们知道cookies是保存在用户本机电脑上的,对服务器带宽没什么影响,但是会有不安全隐患,如果被用户删了怎么办?但是我们还是会经常的使用它的,现在我们看看怎么使用这个cookies。
<%@ page contentType="text/html;charset=UTF-8" language="java" import="java.io.*,javax.servlet.http.Cookie" %>
1
index.jsp
<%
Cookie c = new Cookie("name","zhangsan");
c.setMaxAge(3600);
response.addCookie(c);
%>
<%
Cookie[] cookies = request.getCookies();
for(Cookie a : cookies){
if(a.getName().equals("name")){
out.println(a.getValue());
}
}
%>
设置cookie只需要三个步骤,在上述代码中已经得到体现,第一步,创建cookie对象,没引入cookie包的会报错的,第二步,设置cookie的声明周期,也就是什么时候过期失效,单位毫秒,第三部使用response.addCookie()方法设置cookie。
cookie的读出,我觉的是设计的不合理的地方,它竟然需要将所有的cookie全部读取出来,然后需要遍历找到我们需要的cookie。当然可能.net中的底层也是这么实现的,但是人家提供的接口就很简洁,根据名称就可以直接找到值。
至此,JSP的基本知识就介绍完了,其实还是不全面,但是我会在继续学习中进行补充的。如果文章哪里有错误,欢迎大家指出来。