JSP详解(四大作用域和九大内置对象)
一、JSP
1、什么是JSP?
JSP(Java Server Pages):是以Java语言为基础的动态网页开发技术,
特点:
Servlet特点:在Java源码中嵌入html源码
JSP特点:在html源码中嵌入java代码
JSP就是Servlet
1、tomcat获得JSP文件后,先将JSP转成servlet,变成xxx.java(servlet源码),
D:\\java\\tomcat7.0\\apache-tomcat-7.0.53\\apache-tomcat-7.0.53\\work\\Catalina\\localhost\\test01\\org\\apache\\jsp
tomcat安装目录 引擎 主机 项目 固定包名 这个之下就是存放着jsp变成的servlet文件.java和编译文件.class
2、tomcat将java文件编译成class文件
3、tomcat运行class文件,并将结果输出到浏览器,
实例:
创建一个jsp。查看其转换后的servlet代码。
NewFile.jsp
复制代码
NewFile_jsp.java
复制代码
可以看到public final class NewFile_jsp extends org.apache.jasper.runtime.HttpJspBase implements org.apache.jasper.runtime.JspSourceDependent
NewFile_jsp.java继承自HttpJspBase。来看看HttpJspBase的源码
HttpJspBase.java
复制代码
public abstract class HttpJspBase extends HttpServlet implements HttpJspPage
看到了一个熟悉的类,HttpServlet,我们编写Servlet时就是继承自该类,这里也是继承HttpServlet,并且HttpJspBase的源码会发现,生命周期也是有init()方法,service()方法,destory()方法,相当于_jspService()方法就是servlet的service()方法的执行,所以说JSP也是一个servlet。
我们在JSP写的所有html代码,都会被转换为servlet中的out.write(html)代码来输出。看图
小总结:
对于jsp转换成的servlet源码,之后我们会在进行详细分析,现在只需要知道jsp中的内容在servlet中被转换成什么了,在哪里被转换了即可。其中_jspService()方法的详细内容下面会讲解
注意:jsp 生成java源码,默认第一次生成,之后直接执行,除非内容修改,具体点说,由于JSP只会在客户端第一次请求的时候被编译,因此第一次请求JSP时会感觉比较慢,而之后的请求因为不会编译JSP,所以速度就快多了,如果将Tomcat保存的JSP编译后的class文件删除,Tomcat也会重新编译JSP。在开发Web程序的时候经常需要修改JSP,Tomcat能够自动检测到JSP程序的改动,如果检测到JSP源代码发生了改动,Tomcat会在下次客户端请求JSP时重新编译JSP,而不需要重启Tomcat,这种自动检测功能默认是开启的,检测改动会消耗少量的时间,在部署web应用程序的时候可以在web.xml中将它关掉。这也就是为什么我们能够在jsp页面直接修改内容,而不用重新启动服务器的原因。
因为JSP就是servlet,那么生命周期也就是跟serlvet一样。
JSP和servlet有一点区别就在于:jsp是先部署后编译,而servlet是先编译后部署。
二、JSP语法
JSP模版数据:
就是JSP中的HTML代码,它的内容给是固定的,无论程序如何运行模版数据输出到客户端浏览器时都不会发生改变,当我们创建一个JSP时,模版就已经固定了。
元素:JSP中的java部分,包括脚本(JavaScript,或者java代码)以及JSP指令(Directive)与JSP标签(Tag)等,元素决定着程序的流程,元素是不会显示到浏览器的。这几个都会在接下来讲解到
JSP脚本:
1、使用<% 编写java代码 %>,中间java代码必须遵循Java语法,
为什么能够使用out输出?这里就涉及到了JSP的九大内置对象了,后面会讲解到,到时候回过头来在看看这里,就会知道为什么可以使用。
来看看,jsp变为servlet时的代码是如何编写的。
在JSP中JSP指令(后面会讲): <%@ page language=\”java\” contentType=\”text/html; charset=UTF-8\” pageEncoding=\”UTF-8\”%>
这句代码在servlet就变为了response.setContentType(\”text/html; charset=UTF-8\”);//这句代码的意思就是通知tomcat和浏览器都使用UTF-8码表,并且发送回浏览器的数据类型是text/html。这是有JSP指令中画红色部分代码转变的,而指令中的pageEncoding=\”UTF-8\”的意思是JSP转换为Servlet时采用UTF-8码表编码,因为可能JSP中包含中文。
对于JSP模版数据来说,就原封不动的使用out.write()来输出到浏览器
而对于JSP元素来说,就会变成普通的java代码,因为在servlet中,就可以直接编写java代码。
2、使用<%=xxx %>来输出结果
使用<%=result %>来输出结果,servlet中就会将其转换为out.print(result)进行输出。输出各种类型数据:int、double、boolean、String、Object等
3、JSP注释
<%– –%> :jsp注释,
// :java单行注释
/* */ :Java多行注释
<!– –> :这个注释,会发送到浏览器端的源码中显示
注释分别在servlet中如何显示的?
JSP注释不会在servlet文件中显示,而java注释则会,但其所有的注释到了浏览器端,都不会出现在源码中,只有<!– –>这个注释会到浏览器的网页源码中去
4、JSP中申明方法与属性(全局变量) 使用<%! 方法、属性%>
这个就不演示了,就是在JSP中编写方法或者属性时,使用<%! %>括起来。
5、在JSP中使用if语句,或者使用for循环,whilt循环等都可以实现,也就是编写脚本而已。
三、JSP指令
指令用来申明JSP页面的一些属性,比如编码方式,文档类型。我们在servlet中也会申明我们使用的编码方式和响应的文档类型的,而JSP就是用指令来申明。上面我们也说到了一条指令,也就是page指令,
JSP指令格式:<%@ directive {attribute=value}* %>
解释:directive:指令名称,例如page指令
attribute=value:紧跟指令名称后面的就是各种属性,以键值对的形式书写
*:代表后面能跟0个或多个属性。
例如: page指令:用来声明JSP页面的属性等。
<%@ page language=\”java\” contentType=\”text/html; charset=UTF-8\” pageEncoding=\”UTF-8\”%> page指令,后面跟着三个属性,分别是language、contentType、pageEncoding。这只是其中的几个属性,并没有写全,page指令允许的属性如下表所示
属性名称 取值范围 描述
language java 解释该JSP文件时采用的语言,一般为java语言,默认为java
extends 任何类的全名 编译该JSP文件时继承哪个类,JSP为Servlet,因此当指明继承普通类时需要实现Servlet的init、destroy等方法
import 任何包名、类名 引入该JSP中用到的类、包等,import是唯一可以声明多次的page指令属性,一个import可以引用uogelei,中间用英文逗号隔开,
如<%@ page import=\”java.util.List,java.util.ArrayList\”%>
session true、false 该JSP内是否内置Session对象,如果为true,则内置Session对象,可直接使用,否则反之,默认为true
autoFlush true,false 是否运行缓存,如果为true,则使用out.println()等方法输出的字符串并不是立刻到达客户端服务器的,而是暂时存到缓存里,缓存了或者程序执行完毕或者执行out.flush()操作时才到客户端,默认为true。
buffer none或者数字KB 指定缓存大小,当autoFlush设为true时有效,例如<%@ page buffer=10kb%>
isThreadSafe true,false 是否线程安全,如果为true,则运行多个线程同时运行该jsp程序,否则只运行一个线程,其余线程等待,默认为false
isErrorPage true,false 指定该页面是否为错误显示页面,如果为true,则该JSP内置有一个Exception对象exception,可直接使用,否则没有,默认为false
errorPage 某个JSP页面的相对路径 指明一个错误页面,如果该JSP程序抛出一个未捕捉的异常,则转到errorPage指定的页面,errorPage指定的页面通常isErrorPage属性为true,且内置的exception对象为未捕捉的异常
contentType 有效的文档类型 客户端浏览器根据该属性判断文档类型,例如 HTML格式为text/html、纯文本格式为text/plain、JPG图像为image/jpeg、GIF图 像为image/gif、WORD文档为application/msword,该属性常跟着charset设置编码一起,作用是通知服务器和浏览器都使用同一 个码表
info 任意字符串 指明JSP的信息,该信息可以通过Servlet.getServletInfo()方法获取到
trimDirective Whitespaces true、false 是否去掉指令前后的空白字符,默认为false
pageEncoding UTF-8,ISO-8859-1等 指定一张码表来对该JSP页面进行编码,include指令
比较简单,只有一种形式 <%@ include file=\”relativeURL\”%> relativeURL:本应用程序内另一个JSP文件或者HTML文件的路径,例如,网址内所有页面均有一个统一风格的导航栏和页脚版权,那么就可以使用该指令将其包含进来,
特点:include指令会将包含页面的源代码添加到使用include指令的页面中来,然后编译成class文件,而等下会讲到的一个JSP行为,<jsp:include page=\”relativeURL\”>作用跟include指令一样,但是不同的是,include行为是运行时单独执行包含页面,然后把执行的结果包含到本页面来,属于先运行后包含。
taglib指令 JSP支持标签技术,后面会讲到标签的用法,jstl标签库的使用等,作用:用来指明JSP页面内使用的JSP标签库,taglib指令有两个属性,uri为类库的地址,prefix为标签的前缀<%@ taglib uri=\”http://java.sun.com/jsp/jstl/core\” prefix=\”c\”%>
四、JSP行为
前面讲了JSP语法,介绍了JSP页面中的内容有哪些,分别有什么作用,就两个东西,模块数据和元素。其中元素有包括脚本,指令,标签,脚本就是JSP中嵌入java代码,指令作用就是申明页面的属性,那标签是干嘛的,标签分为JSP自带内置的标签,和通过taglib指令来使用JSP标签库,或者自定义标签。现在我们先来讲一些JSP内置的标签。
JSP内置的标签就被称为JSP行为(JSP Actions)。只要书写很少的标记代码就能使用JSP提供的丰富功能,JSP行为其实是对常用的JSP功能的抽象与封装,可以取代jsp脚本,让JSP中就少一些嵌入java代码的地方。
格式:<jsp:elements {attribute=\”value\”}* />
jsp:标签的前缀,说明是jsp内置的标签,
elements:行为的名称,
attribute=value:使用键值对来编写属性
*:能指定0个或多个属性对
<jsp:include />行为
include行为用于运行时包含某个文件,如果被包含的文件为JSP程序,则先会执行JSP程序,然后在把执行的结果包含进来。 作用是跟include指令一样的,唯一的区别就在于,include指令是将被包含的文件的源码加入到了本JSP程序中,然后在进行编译,属于静态包含,而include行为只是将被包含的文件的运行结果包含进自己。属于动态包含
Java bean行为
是一组与Java Bean 相关的行为,包括useBean行为、setProperty行为、getProperty行为等
Java Bean就是普通的Java类,也被称为POJO,只有私有的属性与对应的getter方法和setter方法,注意其中当私有的属性为boolean类型时,习惯上一般把getter方法写成isXxx();而不是getXxx();
useBean行为
<jsp:useBean id=\”beanObject\” class=\”className\” scope=\”Value\”> 作用:在jsp中定义一个java bean对象,
id:指明Java Bean对象的名称,JSP中可以使用该名称引用该Java Bean对象,相当于给new出来的对象取一个变量名,
class:Java Bean类的全名
scope:该java bean对象的作用范围,可以写的就四个,也就是JSP的四大作用域,page、request、session、application
page:只能在当前JSP页面使用,如果不在JSP页面,那么就会失效
request:这个前面学过,A页面请求转发到B页面,那么使用的是同一个request,那么A,B页面都算是request的作用域,也就是通过请求转发的页面都是其作 用域
session:该作用域在一个web项目下任何位置应该读访问的到,只要cookie不关闭,并且cookie设置 的访问路径为\”/\”,
application:其实就是Servlet中的servletContext,服务器下的所有项目都能访问到。
setProperty行为
<jsp:setProperty name=\”beanName\” property=\”propertyName\” value=\”\”>
对Java Bean对象进行属性的设置
name:java bean对象的名称,也就是在useBean行为中的id
property:对象中的属性名,
value:要对其属性进行赋值的值
getProperty行为
<jsp:getProperty name=\”beanName\” property=\”propertyName\” />
获取JavaBean对象的某个属性值
name:java bean 对象的名称,也就是在useBean行为中的id
property:对象的属性名
例子: javabean:User.java NewFile.jsp
User.java
复制代码
NewFile.jsp
复制代码
可以查看NewFile.jsp变为servlet后的源代码,看看我们写的javabean行为会被转换为何种语句
这里出现了一个JSP九大内置对象中的一个,pageContext。现在简单提一句,pageContext就是JSP页面的管理者(上下文),其中的getAttribute(name,scope)方法是获取指定作用域中的数据的,如果getAttribute(name)方法的话,默认是对page作用域进行操作,findAttribute(name)依次从page、request、session、application获得内容。
在第一个红框中,就代表中我们的useBean行为,其中进行了一次判断,就是如果在page作用域中找不到user这个对象,那么就创建一个新的,否则就使用找到的这个user对象,
第二个红框中,代表着我们的setProperty行为,先找到user对象,然后在对其属性进行赋值
第三个红框中,代表着我们的getProperty行为,也是先找到user对象,然后在获取其属性的值。
注意:对于javabean行为来说,有一个特点的地方,就是当请求过来的参数对应javabean的属性时,可以为其一次性设置所有的值
<jsp:setProperty name=\”user\” property=\”*\” /> //设置user的所有属性,属性值从request中自动取得,*代表所有属性。
<jsp:forward />行为
实现请求转发功能,Servlet中通过request.getRequestDispatcher(\”someServlet\”).forward(request,response);而在JSP中也能够实现相同的功能,只不过用的是<jsp:forward />行为,实际上forward行为就是对其进行了封装。
格式:
<jsp:forward page=\”someServlet\”>
<jsp:param name=\”param1\” value=\”value1\”/>
<jsp:param name=\”param2\” value=\”value2\”/>
</jsp:forward>
page:需要跳转到的页面或者servlet、 <jsp:param/>参数行为,带一些参数过去,name、value是以键值对的形式带过去的
例如:
NewFile.jsp
MyServlet.java
访问:http://localhost:8080/test01/NewFile.jsp
浏览器地址栏没有改变,说明是请求转发
NewFile_jsp.java
使用return的好处是执行完上面的转发,就直接return,没有必要在执行下面的代码了,对参数使用了URLEncode进行编码,说明该<jsp:param/>可以直接传递中文,但是前提是要设置request.setCharacterEncoding(\”UTF-8\”);为什么这样做,看上面框起来中的代码。
<jsp:directive/>行为
directive行为,就相当于JSP指令,比如<jsp:directive.page/>相当于<%@ page %>指令,等等其它指令是一样的书写格式。
五、JSP隐藏的九大内置对象和四大作用域
之前那么一大段内容,就是为了讲解什么是JSP,JSP中的内容是怎么样的,到这里应该知道的差不多了, 但我们还要深入了解一些东西,我们知道JSP中的内容就只有两种,模版数据,和元素,元素就包括了指令呀,脚本呀,标签(行为)呀,脚本会慢慢被标签全部代替,也就是说JSP中基本上不会嵌入Java代码,但是我们也知道JSP会转换为servlet,在Servlet中,输出数据时,都需要通过response.getWrite();但是在JSP中,直接使用out对象进行输出,为什么呢?这就是因为out为JSP的一个隐藏对象,JSP中内置了9个隐藏对象,使得JSP比Servlet使用起来更简单,更方便,
page、config、application、request、response、session、out、exception、pageContext
page:page对象代表当前JSP页面,是当前JSP编译后的Servlet类的对象。相当于this。
config:标识Servlet配置,类型:ServletConfig,api跟Servlet中的ServletConfig对象是一样的,能获取该servlet的一些配置信息,能够获取ServletContext
application:标识web应用上下文,类型:ServletContext,详情就看Servlet中的ServletContext的使用
request:请求对象, 类型:httpServletRequest
response:响应对象 类型:httpServletResponse
session:表示一次会话,在服务器端记录用户状信息的技术
out:输出响应体 类型:JspWriter
exception 表示发生异常对象,类型 Throwable,在上面我们介绍page指令中的一个errorPage属性时就有说到他
pageContext:表示 jsp页面上下文(jsp管理者) 类型:PageContext
标记了红色的对象就是JSP独有的,其他的都是Servlet中的老东西。
在这个由jsp转换为servlet的文件中,只能看到8个内置对象,少了exception对象,因为我们在将page指令时,说过一个isErrorPage属性,默认是false,被关闭了,所以其中并没有exception对象。
JSP的四大作用域:page、request、session、application
这四大作用域,其实就是其九大内置对象中的四个,为什么说他们也是JSP的四大作用域呢?因为这四个对象都能存储数据,比如request.setAttribute()注意和request.setParameter()区分开来,一个是存储在域中的、一个是请求参数,session.setAttribute()、application其实就是SerlvetContext,自然也有setAttribute()方法。而page作用域的操作就需要依靠pageContext对象来进行了。在上面我们也有提到JSP的四大作用域,
page作用域:代表变量只能在当前页面上生效
request:代表变量能在一次请求中生效,一次请求可能包含一个页面,也可能包含多个页面,比如页面A请求转发到页面B
session:代表变量能在一次会话中生效,基本上就是能在web项目下都有效,session的使用也跟cookie有很大的关系。一般来说,只要浏览器不关闭,cookie就会一直生效,cookie生效,session的使用就不会受到影响。
application:代表变量能一个应用下(多个会话),在服务器下的多个项目之间都能够使用。比如baidu、wenku等共享帐号。
out对象:
类型:JspWriter
jsp 输出底层使用 response.getWriter();什么意思呢?这里就要讲解一下JSP缓存和Servlet缓存了,输出的过程是这样的
JSP页面转换为Servlet后,使用的out对象是JspWriter类型的,所以是会先将要发送的数据存入JSP输出缓存中,然后,等JSP输出缓存满了在自动刷新到servlet输出缓存,等serlvet输出缓存满了,或者程序结束了,就会将其输出到浏览器上。除非手动out.flush()。
验证servlet输出缓存和JSP输出缓存和我们上面所说的是正确的。
4、提供作用域常量
pageContext对象:重点
这个功能就比较强大了,比较牛逼,基本上什么他都有,因为是它是JSP页面的管理者(上下文),所以JSP中的内置对象呀,它统统能够获得,下面介绍它的api。
1、获得其它八大内置对象 getXxx()
pageContext.getOut(); //获得out对象
pageContext.getApplication(); //获得application对象
等等….
2、对作用域的属性进行操作(四大作用域)
对默认作用域的属性进行操作。page
pageContext.getAttribute(name); //获得page作用域数据
pageContext.setAttribute(name,value); //给page作用域设置内容
pageContext.removeAttribute(name); //给page作用域移除内容
3、对指定作用域的属性进行操作
getAttribute(name,scope); //获得 指定作用域中的数据
setAttribute(name,value); //给指定作用域设置内容
removeAttribute(name ,scope) 移除指定作用域的内容(page/request/session/application)
4、提供作用域常量
PageContext.PAGE_SCOPE page
PageContext.REQUEST_SCOPE request
PageContext.SESSION_SCOPE response
PageContext.APPLICATION_SCOPE application
5、一次获得指定名称内容
findAttribute(name); //依次从page、request、session、application 获得内容
response对象:
就是响应对象,、如果不了解就看看讲解request和response的这一章节的内容
config对象:
类型:ServletConfig
能够获取servlet的初始化参数,获取servletContext对象,获取servletName
api详情请看讲解servlet这一章节
exception异常对象:
包含了异常的信息
使用它,必须结合page指令中的isErrorPage属性和errorPage属性。
exception.jsp 抛异常的一个NullPointException,并且跳转到error.jsp错误显示页面 errorPage属性的意思是如果发生未捕捉到的异常,将会跳转到error.jsp页面
error.jsp isErrorPage属性说明该页面是一个错误显示页面,则可以使用exception对象
访问http://localhost:8080/test01/exception.jsp
六、总结
通过上面一大堆一大堆的学习,我们应该知道这些东西
1、什么是JSP?
JSP本质上就是一个servlet,因为servlet输出html太麻烦了,所以就有了JSP,JSP就是专门用来书写html的,当然其中也能写java代码。
2、JSP的内容包括什么?
模版数据和元素。其中元素有包括脚本(java代码)、指令(页面属性)、和行为(标签,为了JSP中不嵌入那么多java代码衍生的)
3、JSP中九大内置对象是哪九个?
九大内置对象,page、config、appliction、request、response、session、out、exception、pageContext
4、九大内置对象和servlet中对象的关系
page就是jsp转换为servletservlet对象本身,也就是this
config — Servlet中的servletConfig
application — Servlet中的ServletContext
request – Servlet中的request
response – Servlet中的response
session – Servlet中的session
out – JspWriter
exception – 异常对象
pageContext – 表示 jsp页面上下文(jsp管理者) 类型:PageContext,
其中pageContext对象最牛逼,有了他就拥有了天下,哈哈~
5、JSP中的四大作用域。
page、request、session、application
其中操作page域中属性需要借助pageContext对象。
6、JSP中还有其他两大块内容
一个是EL表达式,很重要,
另一个是jstl标签库的使用,也很重要
rvlet中的session
out – JspWriter
exception – 异常对象
pageContext – 表示 jsp页面上下文(jsp管理者) 类型:PageContext,
其中pageContext对象最牛逼,有了他就拥有了天下,哈哈~
5、JSP中的四大作用域。
page、request、session、application
其中操作page域中属性需要借助pageContext对象。
6、JSP中还有其他两大块内容
一个是EL表达式,很重要,
另一个是jstl标签库的使用,也很重要
JSP的基本使用总结
- JSP的全称是Java Server Pages,即Java的服务器页面
- JSP的主要作用是代替Servlet程序回传HTML页面的数据
- web目录(或其他)右击 –> new –> JSP/JSPX –> 输入文件名 –> 选择JSP file创建
JSP页面本质上是一个Servlet程序,第一次访问JSP页面时(运行Tomcat服务器后在浏览器地址栏输入路径),Tomcat服务器会将此JSP页面翻译成为一个Java源文件,并对其进行编译成为.class字节码文件(一个.java,一个.class),当打开.java文件时发现其中的内容是:
而HttpJspBase类直接继承于HttpServlet类,即JSP翻译出来的Java类间接继承于HttpServlet类,证明JSP页面是一个Servlet程序
JSP头部的page指令:
JSP头部的page指令可以修改JSP页面中的一些重要属性或行为
(以下属性均写在page指令中,默认page指令中没有出现的属性都采用默认值):
(1) contentType属性:表示JSP返回的数据类型是什么,即response.setContentType()的参数值
(2) language属性:表示JSP翻译之后是什么语言文件(目前只支持Java)
(3) pageEncoding属性:表示当前JSP文件本身的字符集(可在IDEA右下角看到)
(4) import属性:表示导包(导类),与Java一致
(5) autoFlush属性:设置当out输出流缓冲区满了之后是否自动刷新缓冲区,默认值是true
(6) buffer属性:设置out缓冲区的大小,默认是8kb
注意:out缓冲区满了之后不能自动刷新的话会报错
(7) errorPage属性:设置当JSP页面运行出错时自动跳转到的页面(错误信息页面)的路径,这个 路径一般都是以斜杠打头,表示请求的地址是http://ip:port/工程路径/,对应代码web目录
(8) isErrorPage属性:设置当前JSP页面是否是错误信息页面,默认是false,如果是true可以 获取错误信息
(9) session属性:设置访问当前JSP页面时是否会创建HttpSession对象,默认值是true
(10) extends属性:设置JSP页面翻译出来的Java类默认继承谁
注意:以上默认值除非有特殊需要,否则不建议修改
格式:<%! 声明Java代码 %>
作用:可以给JSP翻译出来的Java类定义属性、方法、静态代码块、内部类等
特点:不会在浏览器的页面上显示出来,仅存在于翻译后的Java类中
代码演示:声明脚本的使用(此JSP文件在web目录下,名为First.jsp)
%@ page contentType=\”text/html;charset=UTF-8\” language=\”java\” %>
<%@ page import=\”java.util.HashMap\” %>
<%@ page import=\”java.util.Map\” %>
<html>
<head>
<title>Title</title>
</head>
<body>
<%–1.声明类属性–%>
<%!
private String name;
private static Map<String, Object> map;
%>
<%–2.声明类方法–%>
<%!
public int sum() {
return 12;
}
%>
<%–3.声明静态代码块–%>
<%!
static {
map = new HashMap<String, Object>();
map.put(\”key1\”, \”value1\”);
}
%>
</body>
</html>
对应的翻译后的java源文件:
格式:<%=表达式 %>
作用:在浏览器的JSP页面上输出数据(只有此脚本可以在浏览器的页面上输出数据)
特点:
(1) 所有的表达式脚本都会被翻译到对应的Java类的_jspService()方法中,故表达式脚本可以 直接使用_jspService()方法参数中的对象
(2) 表达式脚本都会被编译后的Java类中的out.print()方法输出到浏览器页面上
(3) 表达式脚本中的表达式不能以分号结束
代码演示:表达式脚本的使用(此JSP文件在web目录下,名为First.jsp)
<%=22 %> <br/>
<%=\”可以输出字符串\” %> <br/>
<%=map %> <br/>
<%–使用_jspService方法中的对象–%>
<%=request.getParameter(\”username\”) %>
启动Tomcat服务器后浏览器的运行结果:
对应的翻译后的Java源文件(在_jspService方法中):
注意:
1.write方法中的标签、转义字符自动识别为对应的
功能,不在页面输出,执行各自代表的功能
2.out的两个方法也在_jspService方法中,也都是java语言
3.只有print、write方法、表达式脚本中的内容才可在浏览器中显示,其余Java代码的sout在控制台输出
格式:<% Java语句 %>
作用:在JSP页面中可以编写需要的Java代码
特点:
(1) 代码脚本翻译后都在_jspService方法中,故代码脚本可以直接使用此方法参数中的对象
(2) 可以由多个代码脚本块组合完成一个完整的Java语句
(3) 代码脚本还可以和表达式脚本一起组合使用,在JSP页面上输出数据
代码演示:代码脚本的使用(此JSP文件在web目录下,名为First.jsp)
<%–1.if语句–%>
<%
int i = 1;
if (i == 1) {
System.out.println(\”我爱祖国!\”);
} else {
System.out.println(\”我很爱祖国!\”);
}
%> <br/>
<%–2.for循环语句–%>
<%
for (int j = 0 ; j < 3; j++) {
System.out.println(\”第\” + j + \”次循环\”);
}
%> <br/>
<%–3.使用_jspService方法参数中的对象–%>
<%
String username = request.getParameter(\”username\”);
System.out.println(\”username对应的值为:\” + username);
%>
运行结果:
启动Tomcat服务器后在地址栏输入:http://localhost:8080/MyTest/First.jsp?username=Jaychou
在控制台显示:
对应的翻译后的Java源文件(在_jspService方法中):
1.HTML注释:<!–HTML注释–>
HTML注释会被翻译到JSP文件对应的Java类的_jspService方法中,以out.write()输出到客户端,
write方法会自动识别标签,执行标签对应的功能,不会在浏览器的页面上输出注释
2.Java注释:(1) //单行注释 (2) /*多行注释*/
Java注释要写在声明脚本和代码脚本中才被认为是Java注释,会被翻译到JSP文件对应的Java类的_jspService方法中,在对应的Java类中也是注释
3.JSP注释:<%- -这是JSP注释- -%>
JSP注释中的内容不会在JSP文件翻译后的Java类中出现,即注释中的内容没有任何功能
JSP的内置对象指的是Tomcat服务器将JSP页面翻译为Java类之后内部提供的九大对象:
(将page指令的isErrorPage属性写成true可以出现exception对象)
request:请求对象
response:响应对象
pageContext:JSP的上下文对象
session:会话对象
application:ServletContext对象
config:ServletConfig对象
out:JSP输出流对象
page:指向当前JSP的对象
exception:异常对象
域对象是指可以像Map一样存取数据的对象,四个域对象功能一样,只是对数据的存取范围不同
代码演示1:四个域对象存取数据的范围的不同(在web目录下创建scope1.jsp)
<%@ page contentType=\”text/html;charset=UTF-8\” language=\”java\” %>
<html>
<head>
<title>scope1</title>
</head>
<body>
<h1>scope1.jsp页面</h1>
<%
//向四个域对象中分别保存数据
pageContext.setAttribute(\”key\”, \”pageContext\”);
request.setAttribute(\”key\”, \”request\”);
session.setAttribute(\”key\”, \”session\”);
application.setAttribute(\”key\”, \”application\”);
%>
<%– <jsp:forward page=\”\”></jsp:forward>是请求转发标签,
page属性设置请求转发的路径 –%>
<jsp:forward page=\”/scope2.jsp\”></jsp:forward>
</body>
代码演示2:在web目录下创建scope2.jsp
<head>
<title>Title</title>
</head>
<body>
<h1>scope2.jsp页面</h1>
<%– JSP页面中不加任何标签直接输入的内容被write方法输出在浏览器的页面上 –%>
pageContext域是否有值:<%=pageContext.getAttribute(\”key\”)%> <br>
request域是否有值:<%=request.getAttribute(\”key\”)%> <br>
session域是否有值:<%=session.getAttribute(\”key\”)%> <br>
application域是否有值:<%=application.getAttribute(\”key\”)%> <br>
</body>
运行结果1:
运行结果2:
注意:若四个域对象在使用时范围都可满足要求,则使用的优先顺序是(范围从小到大):
pageContext –> request –> session –> application
1.相同点:response表示响应,用于给客户端(浏览器)返回内容
out同样也是用于给客户端(浏览器)输出内容
2.不同点:
3.注意:由于官方的代码中翻译后的Java代码底层都是使用out进行输出,故一般都使用out进行 输出,out又分为write方法和print方法:
(1) out.print():会将任何内容转换成字符串后调用write方法输出
(2) out.write():输出字符串没有问题,但输出int型时会将int转换成char输出,导致输出的并非是想要的数字而是数字对应的ASCII码
结论:JSP页面的代码脚本中任何要输出在浏览器的内容均使用out.print()方法
(1)使用场景:
(2)使用方法:
<%@include file=\”\”%>
其中file属性设置要包含的JSP页面,以/打头,代表http://ip:port/工程路径/,对应web目录
代码演示1:在web目录下创建body.jsp
<body>
头部信息 <br>
主体信息 <br>
<%@include file=\”/foot.jsp\”%>
</body>
代码演示2:在web目录下创建foot.jsp
<body>
页脚信息 <br>
</body>
运行结果:
(3)静态包含的特点:
①静态包含不会将被包含的JSP页面翻译成.java.class文件
②静态包含是把被包含的页面的代码拷贝到body.jsp对应的Java文件的对应位置执行输出
(1)使用方法:
<jsp:include page=””></jsp:include>
其中page属性设置要包含的JSP页面,与静态包含一致
(2)动态包含的特点:
①动态包含将被包含的JSP页面翻译成.java.class文件
②动态包含还可以传递参数
③动态包含底层使用如下代码调用被包含的JSP页面执行输出:
org.apache.jasper.runtime.JspRuntimeLibrary.include(request, response, “/foot.jsp”, out, false);
代码演示1:在web目录下创建body.jsp
<body>
头部信息 <br>
主体信息 <br>
<jsp:include page=\”/foot.jsp\”>
<jsp:param name=\”username\” value=\”Jaychou\”/>
<jsp:param name=\”password\” value=\”root\”/>
</jsp:include>
</body>
注意:
- 设置参数的标签要写在动态包含之中
- 出现Expecting “jsp:param” standard action with “name” and “value” attributes异常,两个原因:
①动态包含中未设置参数但没有把<jsp:include page=””></jsp:include>放在一行上
②动态包含中加了注释
代码演示2:在web目录下创建foot.jsp
<body>
页脚信息 <br>
<%=request.getParameter(\”username\”)%>
</body>
运行结果:
(3)动态包含的底层原理:
(1) Listener监听器是JavaWeb的三大组件之一
(2) Listener监听器是JavaEE的规范(接口)
(3) Listener监听器的作用是监听某件事物的变化,然后通过回调函数反馈给程序做一些处理
ServletContextListener监听器可以监听ServletContext对象的创建和销毁(web工程启动时创建,停止时销毁),监听到创建和销毁之后都会调用ServletContextListener监听器的方法进行反馈:
public interface ServletContextListener extends EventListener {
//在ServletContext对象创建之后调用
public void contextInitialized(ServletContextEvent sce);
//在ServletContext对象销毁之后调用
public void contextDestroyed(ServletContextEvent sce);
}
(1) 编写一个类实现ServletContextListener接口
(2) 重写两个方法
(3) 在web.xml文件中配置监听器
代码演示1:创建一个类
ublic class ListenerTest implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent servletContextEvent) {
System.out.println(\”ServletContext对象创建\”);
}
@Override
public void contextDestroyed(ServletContextEvent servletContextEvent) {
System.out.println(\”ServletContext对象销毁\”);
}
}
代码演示2:在web.xml中配置
<listener>
<!– <listener-class>标签中写上述程序的全类名 –>
<listener-class>com.qizegao.servlet.ListenerTest</listener-class>
</listener>
运行结果:
Tomcat服务器启动之后控制台输出ServletContext对象创建
Tomcat服务器停止之后控制台输出ServletContext对象销毁
注意:
- 查看翻译后的Java源文件的方法:启动Tomcat服务器访问到JSP页面之后在控制台输出的信息的前端找到Using CATALINA_BASE中的路径,在硬盘中打开此目录,点击work –> Catalina –> localhost,找到对应的工程文件夹寻找即可
- 访问JSP页面其实是在执行对应的翻译后的Java代码的_jspService方法:翻译后的Java类中没有service方法,而是重写了父类的_jspService方法,这个方法会被父类的service方法调用
[机智]
jsp原理详解
jsp
1 什么是JSP
Java Server Pages : Java服务器端页面,也和Servlet一样,用于动态web技术!
最大的特点:
写Jsp就像写HTML
区别:
HTML只给用户提供静态的数据
JSP页面中可以嵌套Java代码,为用户提供动态数据
2 jSP原理
思路: JSP到底怎么执行的!
代码层面没有任何问题
服务器内部 工作
tomcat 中有一个work目录
idea中使用tomcat会在idea中产生一个work目录
这个是 idea tomcat的工作空间
路径:
浏览器向服务器发送请求,不管访问什么资源,起始都是在访问Servlet
JSP最终也会被转换成为一个Java类
JSP本质上就是一个Servlet
1.判断请求
2.内置对象
3.输出页面前增加的代码
以上的这些个对象我们可以在JSP页面中直接使用
jsp原理图
在JSP页面中;
只要是Java代码就会原封不动的输出
如果是HTML代码,就会被转换
这样的格式,输出到前端
3 JSP基础语法
任何语言都有自己的语法,Java中有,JSP作为Java技术的一种应用,它拥有一些自己的扩容的语法,Java所有语法都支持
JSP表达式
述(最多18字
jsp 脚本片段
脚本片段的实现
JSP声明
JSP声明: 会被编译到JSP生成Java类中! 其他的,就会被生成到 _jspService 方法中
在JSP中嵌入Java代码即可
jsp的注释,不会在客户端显示,HTML就会
4 JSP指令
总结:
在实际开发中建议使用 下面 jsp:include/的方式
如果使用上面那一种的话 如果在header.jsp或者在footer.jsp中声明了 int a=1; 又在当前jsp中声明了相同的变量 在jsp页面不会报错,去了浏览器里面 就报错了。
所以建议使用 jsp标签 进行页面拼接
5 九大内置对象
>PageContext 存东西
>Request 存东西
>Response
>Session 存东西
>Application 【ServletContext 】 存东西
>config 【ServletConfig】
>out
>page
>exception
equest:客户端向服务器发送请求,产生的数据,用户看完了就没作用了,比如:新闻 用户看完了没作用了。
session:客户端向服务器发送请求,产生的数据,用户用完了一会还有用,比如:购物车;
application:客户端向服务器发送请求,产生的数据,一个用户用完了,其他用户还可能使用:比如:聊天数据
6 JSP标签 JSTL标签 EL表达式
开始之前导入所学的依赖
EL表达式: ${ }
>获取数据
>执行运算
>获取web开发的常用对象
JSP 标签
发送:
接收:
JSTL 表达式
JSTL表达式的使用就是为了弥补HTML标签的不足;它自定义许多标签,供我们使用,标签的功能和Java代码一样
核心标签
JSTL 标签库使用步骤
引入对应的taglib
使用其中的方法
在Tomcat也需要引入jstl的包,否则会报错
c:if
c:choose c:when
c:forEach
本文作者及来源:Renderbus瑞云渲染农场https://www.renderbus.com
文章为作者独立观点不代本网立场,未经允许不得转载。