JSP(Java Server Pages):java服务器页面
它是一个简化的Servlet
特点是在HTML中嵌入java代码
主要用于页面数据的展示
jsp同样是单例多线程的
JSP注释
HTML 注释格式 <!-- -->
JSP的注释格式:<%-- --%>
JSP运行原理
jsp在第一次被访问时会被Web容器Tomcat翻译成servlet的.java文件再编译为.class文件执行
其中jsp页面中的HTML,CSS,JavaScript及普通文本,均会被翻译到的out.write()中
PrintWriter out = response.getWriter();
out.write("\r\n");
out.write("<!DOCTYPE html>\r\n");
out.write("<html lang=\"en\">\r\n");
out.write(" <head>\r\n");
out.write(" <meta charset=\"UTF-8\" />\r\n");
out.write(" <title>");
out.print(request.getServletContext().getServerInfo() );
out.write("</title>\r\n");
out.write(" <link href=\"favicon.ico\" rel=\"icon\" type=\"image/x-icon\" />\r\n");
out.write(" <link href=\"favicon.ico\" rel=\"shortcut icon\" type=\"image/x-icon\" />\r\n");
out.write(" <link href=\"tomcat.css\" rel=\"stylesheet\" type=\"text/css\" />\r\n");
out.write(" </head>\r\n");
out.write("\r\n");
JSP注释在翻译时会被忽视,而HTML 注释会被翻译到out.write输出中
Jsp代码块
1.声明的变量不能添加权限访问控制符的
2.不能定义方法
3.不能定义静态语句块
4.使用格式:<% %>;如:<% system.out.print("123"); %>输出在控制台
5.java代码末尾必须加;
JSP的声明语句块
使用格式:<%! %>;这样声明的变量是成员变量,可以添加访问控制符,可以定义方法,可以定义静态语句块
JSP的表达式块
使用格式<%= %>等价于out.print();它会直接将内容显示到浏览器页面
例如:
<% int count = 5; %> Jsp代码块
<%= count %>页面显示5 Jsp表达式块
JSP的内置对象
JSP的内置对象:在JSP的Java代码块,表达式块等中可以直接使用的引用对象
九大内置对象:
1.pageContext javax.servlet.jsp.PageContext
2.request javax.servlet.http.HttpServletRequest
3.session javax.servlet.htttp.HttpSession
4.application javax.servlet.ServletContext
5.response javax.servlet.http.HttpServletResponse
6.config javax.servlet.ServletConfig代表Servlet的配置信息
7.out javax.servlet.jsp.jspWriter
8.page javax.lang.Object 代表当前Servlet对象
9.exception javax.lang.Throwable
通过pageContext 对象可以获取其它所有的内置对象
域对象
四大域对象:pageContext, HttpRequest,HttpSession,ServletContext
域空间由小到大:pageContext< HttpRequest<HttpSession<ServletContext
域对象共有的方法:
setAttribute(String name, Object o)
getAttribute(String name)
removeAttribute(String name)
域对象—ServletContext
生命周期:当Web应用被加载进容器时创建,代表整个web应用的ServletContext对象,
当服务器关闭或Web应用被移除时,ServletContext对象跟着销毁
作用范围:整个Web应用(应用范围)
作用:
a)在不同Servlet 之间转发
this.getServletContext().getRequestDispatcher("/servlet/Demo10Servlet").forward(request, response);
方法执行结束后,Servlet 就会返回到服务器,再有服务器去调用目标servlet,其中request会重新创建,并将之前的request的数据拷贝进去。
b)读取资源文件。
1、由于相对路径默认相对的是java虚拟机启动的目录,所以我们直接写相对路径将会是相对于tomcat/bin目录,所以是拿不到资源的。
如果写成绝对路径,当项目发布到其他环境时,绝对路径就错了。
2、为了解决这个问题ServletContext提供了:
this.getServletContext().getRealPath("/1.properties"),给一个资源的相对路径,将会返回该资源在当前环境下的绝对路径。
this.getServletContext().getResourceAsStream("/1.properties"),给一个资源的相对路径返回到该资源绝对路径的流。
3、当在非servlet下获取资源文件时,就没有ServletContext对象用了,此时只能用类加载器
classLoader.getResourceAsStream("../../1.properties"),此方法利用类加载器直接将资源加载到内存中,有更新延迟的问题,
以及如果文件太大,占用内存过大。
classLoader.getResource("../1.properties").getPath(),直接返回资源的真实路径,没有更新延迟的问题
域对象—HttpSession域
HttpSession在服务器中,为浏览器创建唯一的内存空间,在其中保存会话相关的信息
1、生命周期:在第一次调用 request.getSession() 方法时,服务器会检查是否已经有对应的session,如果没有就在内存中创建一个session并返回。
a.当一段时间内session没有被使用(默认为30分钟),则服务器会销毁该session。
b.如果服务器非正常关闭(强行关闭),没有到期的session也会跟着销毁。
c.如果调用session提供的invalidate() ,可以立即销毁session。
注意:服务器正常关闭,再启动,Session对象会进行钝化和活化操作。同时如果服务器钝化的时间在session 默认销毁时间之内,
则活化后session还是存在的。否则Session不存在。
如果JavaBean数据在session钝化时,没有实现Serializable 则当Session活化时,会消失。
活化(从硬盘到内存)
钝化(从内存到硬盘)
2、作用范围:一次会话。
域对象—ServletRequest域
1、生命周期:在service 方法调用前由服务器创建,作为service方法的参数。整个请求结束,request生命周期结束。
2、作用范围:(整个请求范围)整个请求链(请求转发也存在)。
当前请求与下一次请求的request域是不同的
3、作用:在整个请求链中共享数据。最常用到:在Servlet 中处理好的数据交给Jsp显示,此时参数就可以放置在Request域中带过去。
域对象—PageContext域
1、生命周期:当对JSP的请求时开始,当响应结束时销毁。
2、作用范围:(页面范围)整个JSP页面,是四大作用域中最小的一个。
作用:
(1)获取其它八大隐式对象,可以认为是一个入口对象。
(2)获取其所有域中的数据
pageContext 操作所有域中存入的数据
public java.lang.Object getAttribute(java.lang.String name,int scope)
public void setAttribute(java.lang.String name, java.lang.Object value,int scope)
public void removeAttribute(java.lang.String name,int scope)
pageContext 获取域中常量
PageContext.APPLICATION_SCOPE
PageContext.SESSION_SCOPE
PageContext.REQUEST_SCOPE
PageContext.PAGE_SCOPE
findAttribute方法,在四大域中搜寻属性,搜寻的顺序是page域、request域、session域、application域,
从小域到大域开始搜索,如果搜索到就直接获取该值,如果所有域中都找不到,返回一个null(与el表达式不同,此处返回null,对网页是不友好的)
(3)跳转到其他资源 其身上提供了forward和include方法,简化重定向和转发的操作
JSP指令
为当前页面做一些基本的属性设置,为页面的运行提供基本的环境
3类指令:
page指令(页面指令)
include指令(包含指令)
taglib指令(标签库指令)
page指令
格式:<%@ page 属性名1= "属性值1" 属性名2= "属性值2" ...%>
page指令(页面指令)的属性:
1.pageEncoding属性:设置字符编码
2.contentType属性:设置字符编码
pageEncoding="utf-8"与contentType="text/html;charset=UTF-8"两者相同都是设置字符编码,这是前者默认是text/html,后者可以任意设置
3.import属性:导入类包
在<%=new Date()%>使用jsp获取时间就需要导包
<%@ page import="java.util.Date , java.text.* "%>导入多个包用逗号隔开
4.errorpage属性:当程序出现错误时跳转的页面,如3 / 0
<%@ page errorpage="/error.jsp" %>这是JSP的动态部分,这是后台路径,参照路径是web应用的根,即WebContent下
5.isErrorPage属性:显示错误信息
<%@ page isErrorPage=true %>
当isErrorPage=true,第九个内置对象exception javax.lang.Throwable才会出现
可以使用如下代码获取错误信息
ex = <%=exception.getMessage()%>
如果是3 / 0 错误就会显示 / by zero
6.session属性:开启或关闭session内置对象
<%@ page session="false"%>关闭内置对象session
但是可以通过request.getSession()获取,因为这个内置对象底层就是通过request来获取的session对象的
<% HttpSession session = request.getSession(false)//有session就用,没有也不创建,request.getSession(true)没有session就创建新的使用
session.getAttribute("user");
%>
include指令
格式:<%@ include file="被包含的文件地址"%>
可以将一个jsp页面包含到另一个jsp页面中
在页面中的多个相同的设置可以编写在include中,里面的变量可以共享,大大减少了工作量
静态联编:获取其它jsp文件的变量或者配置
<%@ include file = "/text.jsp"%>
taglib指令
格式:<%@ taglib uri="标签库地址" prefix="前缀"%>
在jsp页面中引入标签库(jstl标签库、struts2标签库)
例如:<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
获取jstl技术的核心库
JSP动作
使用系统定义好的标签来完成本应该由Java代码完成的功能
为了JSP页面简洁,我们会尽量少使用Java代码块,表达式块
使用格式:<jsp:动作名称 属性名 = 属性值 属性名 = 属性值。。。/ >
常见的jsp动作
forward动作
include动作
这两个动作底层都是使用RequestDispatcher的forward方法与include方法
而这两种请求转发方式的本质区别是:标准输出流的开启时间不一样
forward方法的标准输出流是在目标资源中开启的标准输出流
include方法的标准输出流是在当前发出包含动作的页面中开启的,
所以forward动作的发起页面是无法向标准输出流中写入数据的,
而include动作的发起页面即目标页面中均可向标准输出流中写入数据
这两个动作都具有一个page属性,用于指向要转向的页面
动态联编:
<jsp:forward page = "/next.jsp" />
如果两个文件使用同一变量就使用静态联编;两个使用同名的不同变量用动态联编