#JSP运行原理
- 每个JSP 页面在第一次被访问时,WEB容器都会把请求交给JSP引擎(即一个Java程序)去处理。JSP引擎先将JSP翻译成一个_jspServlet(实质上也是一个servlet) ,然后按照servlet的调用方式进行调用
- 由于JSP第一次访问时会翻译成servlet,所以第一次访问通常会比较慢,但第二次访问,JSP引擎如果发现JSP没有变化,就不再翻译,而是直接调用,所以程序的执行效率不会受到影响
- JSP引擎在调用JSP对应的_jspServlet时,会传递或创建9个与web开发相关的对象供_jspServlet使用。JSP技术的设计者为便于开发人员在编写JSP页面时获得这些web对象的引用,特意定义了9个相应的变量,开发人员在JSP页面中通过这些变量就可以快速获得这9大对象的引用
- 这9个对象分别是哪些,以及作用也是笔试经常考察的知识点
#JSP九大隐式对象
- request
- response
- config
- application
- exception
- session
- page
- out
- pageContext
JSP九大隐式对象,其中七个已经在Servlet中讲解过了,主要学习out与pageContext
###out
- out隐式对象用于向客户端发送文本数据
- out对象是通过调用pageContext对象的getOut方法返回的,其作用和用法与ServletResponse.getWriter方法返回的PrintWriter对象非常相似
- JSP页面中的out隐式对象的类型为JspWriter,JspWriter相当于一种带缓存功能的PrintWriter,设置JSP页面的page指令的buffer属性可以调整它的缓存大小,甚至关闭它的缓存
- 只有向out对象中写入了内容,且满足如下任何一个条件时,out对象才去调用ServletResponse.getWriter方法,并通过该方法返回的PrintWriter对象将out对象的缓冲区中的内容真正写入到Servlet引擎提供的缓冲区中:
- 设置page指令的buffer属性关闭了out对象的缓存功能
- out对象的缓冲区已满
- 整个JSP页面结束
<body><%out.write("hehe");response.getWriter().write("gaga");%></body>
- 上面的代码运行后,会首先输出gaga,然后再输出hehe。这是因为,隐式对象的缓冲区的内容不会直接回写到客户端,而且会跟Response的缓冲区合并在一起(添加到后面),一次回写到客户端。
- 在实际开发中,如果在JSP中需要显示数据,使用隐式对象,不要使用Response输出(JSP模版也存在这个问题)。
- 同时使用out和response.getWriter()输出数据
- demo:使用JSP实现文件的下载
###pageContext
- pageContext对象是JSP技术中最重要的一个对象,它代表JSP页面的运行环境
- 这个对象不仅封装了对其它8大隐式对象的引用,它自身还是一个域对象,可以用来保存数据,并且还可以通过pageContext操作其它三个域
- 使用pageContext可以访问别的三个域以及8大隐式对象,这种用法主要在自定义标签技术中使用
- 通过pageContext对象,可以直接将数据添加到别的三个域中,也可以直接从三个域中获取
- 使用
findAttribute
方法,直接查找各个域中是否有该属性(EL表达式)
- 这个对象还封装了web开发中经常涉及到的一些常用操作,例如引入和跳转其它资源
- pageContext类中定义了forward方法和include方法来分别简化和替代RequestDispatcher.forward方法和include方法
- 方法接收的资源如果以“/”开头, “/”代表当前web应用
#JSP标签(JSP标准动作)
- JSP标签也称之为Jsp Action(JSP动作)元素,它用于在Jsp页面中提供业务逻辑功能,避免在JSP页面中直接编写java代码,造成jsp页面难以维护
JSP常用标签:
<jsp:include>标签
<jsp:forward>标签
<jsp:param>标签
###include标签
<jsp:include>
标签用于把另外一个资源的输出内容插入进当前JSP页面的输出内容之中,这种在JSP页面执行时的引入方式称之为动态引入- 语法:
<jsp:include page="relativeURL | <%=expression%>" flush="true|false" />
- page属性用于指定被引入资源的相对路径,它也可以通过执行一个表达式来获得
- flush属性指定在插入其他资源的输出内容时,是否先将当前JSP页面的已输出的内容刷新到客户端
<jsp:include>
与include指令的比较
<jsp:include>
标签是动态引入,<jsp:include>
标签涉及到的2个JSP页面会被翻译成2个servlet,这2个servlet的内容在执行时进行合并- 而include指令是静态引入,涉及到的2个JSP页面会被翻译成一个servlet,其内容是在源文件级别进行合并
- 不管是
<jsp:include>
标签,还是include指令,它们都会把两个JSP页面内容合并输出,所以这两个页面不要出现重复的HTML全局架构标签,否则输出给客户端的内容将会是一个格式混乱的HTML文档
###<jsp:forward>
标签
<jsp:forward>
标签用于把请求转发给另外一个资源- 语法:
<jsp:forward page="relativeURL | <%=expression%>" />
,- page属性用于指定请求转发到的资源的相对路径,它也可以通过执行一个表达式来获得
<jsp:param>
标签
-
当使用
<jsp:include>
和<jsp:forward>
标签引入或将请求转发给其它资源时,可以使用<jsp:param>
标签向这个资源传递参数 -
语法1:
<jsp:include page="relativeURL | <%=expression%>"> <jsp:param name="parameterName" value="parameterValue|<%= expression %>" /> </jsp:include>
-
语法2:
<jsp:forward page="relativeURL | <%=expression%>"> <jsp:param name="parameterName" value="parameterValue|<%= expression %>" /> </jsp:include>
-
<jsp:param>
标签的name属性用于指定参数名,value属性用于指定参数值。在<jsp:include>
和<jsp:forward>
标签中可以使用多个<jsp:param>
标签来传递多个参数
###映射JSP
<servlet>
<servlet-name>SimpleJspServlet</servlet-name>
<jsp-file>/jsp/simple.jsp</jsp-file>
<load-on-startup>1</load-on-startup >
</servlet>
<servlet-mapping>
<servlet-name>SimpleJspServlet</servlet-name>
<url-pattern>/xxx/yyy.html</url-pattern>
</servlet-mapping>
###如何查找JSP页面中的错误
- JSP页面中的JSP语法格式有问题,导致其不能被翻译成Servlet源文件,JSP引擎将提示这类错误发生在JSP页面中的位置(行和列)以及相关信息
- JSP页面中的JSP语法格式没有问题,但被翻译成的Servlet源文件中出现了Java语法问题,导致JSP页面翻译成的Servlet源文件不能通过编译,JSP引擎也将提示这类错误发生在JSP页面中的位置(行和列)以及相关信息
- JSP页面翻译成的Servlet程序在运行时出现异常,这与普通Java程序的运行时错误完全一样,Java虚拟机将提示错误发生在Servlet源文件中的位置(行和列)以及相关信息
###重点
- 到此为止,web开发接触到了4个域对象,这4个域对象是学习web的重点,也是笔试经常考察的知识点
- pageContext(称之为page域)
- request(称之为request域)
- session(称之为session域)
- servletContext(称之为application域)
- 明确如下问题
- 什么是域?
- 这4个对象的生命周期?
- 哪种情况下用哪种域对象
###JavaBean相关标签
-
JavaBean是一个遵循特定写法的Java类,它通常具有如下特点
- 这个Java类必须具有一个无参的构造函数
- 属性必须私有化
- 私有化的属性必须通过public类型的方法暴露给其它程序,并且方法的命名也必须遵守一定的命名规范
-
JavaBean在J2EE开发中,通常用于封装数据,对于遵循以上写法的JavaBean组件,其它程序可以通过反射技术实例化JavaBean对象,并且通过反射那些遵守命名规范的方法,从而获知JavaBean的属性,进而调用其属性保存数据
-
Javabean的属性
- JavaBean的属性可以是任意类型,并且一个JavaBean可以有多个属性。每个属性通常都需要具有相应的setter、 getter方法,setter方法称为属性修改器,getter方法称为属性访问器
- 属性修改器必须以小写的set前缀开始,后跟属性名,且属性名的第一个字母要改为大写,例如,name属性的修改器名称为setName,password属性的修改器名称为setPassword
- 属性访问器通常以小写的get前缀开始,后跟属性名,且属性名的第一个字母也要改为大写,例如,name属性的访问器名称为getName,password属性的访问器名称为getPassword
- 一个JavaBean的某个属性也可以只有set方法或get方法,这样的属性通常也称之为只写、只读属性
-
JSP中的JavaBean
- JSP技术提供了三个关于JavaBean组件的动作元素,即JSP标签,它们分别为:
<jsp:useBean>
标签:用于在JSP页面中查找或实例化一个JavaBean组件<jsp:setProperty>
标签:用于在JSP页面中设置一个JavaBean组件的属性<jsp:getProperty>
标签:用于在JSP页面中获取一个JavaBean组件的属性
-
<jsp:useBean>标签
<jsp:useBean>
标签用于在指定的域范围内查找指定名称的JavaBean对象- 如果存在则直接返回该JavaBean对象的引用
- 如果不存在则实例化一个新的JavaBean对象并将它以指定的名称存储到指定的域范围中
- 常用语法:
<jsp:useBean id="beanName" class="package.class" scope="page|request|session|application"/>
- id属性用于指定JavaBean实例对象的引用名称和其存储在域范围中的名称
- class属性用于指定JavaBean的完整类名(即必须带有包名)
- scope属性用于指定JavaBean实例对象所存储的域范围,其取值只能是page、request、session和application等四个值中的一个,其默认值是page
-
<jsp:useBean>
执行原理<jsp:useBean id="currentDate" class="java.util.Date"/>
- 查看翻译后的Servlet源码
-
带标签体的jsp:useBean标签
- 语法:
<jsp:useBean ...> Body </jsp:useBean>
- 功能:Body部分的内容只在jsp:useBean标签创建JavaBean的实例对象时才执行
- 语法:
-
<jsp:setProperty>
标签<jsp:setProperty>
标签用于设置和访问JavaBean对象的属性。- 语法格式:
<jsp:setProperty name="beanName" { property="propertyName" value="{string | <%= expression %>}" | property="propertyName" [ param="parameterName" ] | property= "*" }/>
- name属性用于指定JavaBean对象的名称。
- property属性用于指定JavaBean实例对象的属性名
- value属性用于指定JavaBean对象的某个属性的值,value的值可以是字符串,也可以是表达式。为字符串时,该值会自动转化为JavaBean属性相应的类型,如果value的值是一个表达式,那么该表达式的计算结果必须与所要设置的JavaBean属性的类型一致
- param属性用于将JavaBean实例对象的某个属性值设置为一个请求参数值,该属性值同样会自动转换成要设置的JavaBean属性的类型
-
<jsp:getProperty>
标签<jsp:getProperty>
标签用于读取JavaBean对象的属性,也就是调用JavaBean对象的getter方法,然后将读取的属性值转换成字符串后插入进输出的响应正文中- 语法:
<jsp:getProperty name="beanInstanceName" property="PropertyName" />
- name属性用于指定JavaBean实例对象的名称,其值应与jsp:useBean标签的id属性值相同
- property属性用于指定JavaBean实例对象的属性名
- 如果一个JavaBean实例对象的某个属性的值为null,那么,使用jsp:getProperty标签输出该属性的结果将是一个内容为“null”的字符串