JSP其实是一个Servlet,JSP页面在JSP容器内运行。
第一次请求一个JSP页面时,Servlet/JSP容器要做两件事:
对于同一个JSP页面的后续请求,Servlet容器会查看这个JSP页面从最后一次转换以来是否修改过。如果修改过,那么重新转换和编译执行。如果没有救说明存在该JSP的Servlet了。第一次调用JSP页面的时间总是比后续的更长,因为需要转换和编译。对此可以采用以下优化措施:
JSP API 包含4个包:
JSP页面可以包含模版数据和句法元素,模版元素就是页面内的html标签和文本,而句法则是jsp页面内独有的语法元素。
比如一个简单的index.jsp页面:
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>JSP compiledtitle>
head>
<body>Hello World
body>
html>
则会被转换成叫index_jsp.java根据Servlet容器不同起名有差异
*
* Generated by the Jasper component of Apache Tomcat
* Version: Apache Tomcat/8.0.30
* Generated at: 2016-08-11 21:05:20 UTC
* Note: The last modified time of this file was set to
* the last modified time of the source file after
* generation to assist with modification tracking.
*/
package org.apache.jsp;
import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.jsp.*;
public final class index_jsp extends org.apache.jasper.runtime.HttpJspBase
implements org.apache.jasper.runtime.JspSourceDependent,
org.apache.jasper.runtime.JspSourceImports {
private static final javax.servlet.jsp.JspFactory _jspxFactory =
javax.servlet.jsp.JspFactory.getDefaultFactory();
private static java.util.Map.lang.String,java.lang.Long> _jspx_dependants;
private static final java.util.Set.lang.String> _jspx_imports_packages;
private static final java.util.Set.lang.String> _jspx_imports_classes;
static {
_jspx_imports_packages = new java.util.HashSet<>();
_jspx_imports_packages.add("javax.servlet");
_jspx_imports_packages.add("javax.servlet.http");
_jspx_imports_packages.add("javax.servlet.jsp");
_jspx_imports_classes = null;
}
private volatile javax.el.ExpressionFactory _el_expressionfactory;
private volatile org.apache.tomcat.InstanceManager _jsp_instancemanager;
public java.util.Map.lang.String,java.lang.Long> getDependants() {
return _jspx_dependants;
}
public java.util.Set.lang.String> getPackageImports() {
return _jspx_imports_packages;
}
public java.util.Set.lang.String> getClassImports() {
return _jspx_imports_classes;
}
public javax.el.ExpressionFactory _jsp_getExpressionFactory() {
if (_el_expressionfactory == null) {
synchronized (this) {
if (_el_expressionfactory == null) {
_el_expressionfactory = _jspxFactory.getJspApplicationContext(getServletConfig().getServletContext()).getExpressionFactory();
}
}
}
return _el_expressionfactory;
}
public org.apache.tomcat.InstanceManager _jsp_getInstanceManager() {
if (_jsp_instancemanager == null) {
synchronized (this) {
if (_jsp_instancemanager == null) {
_jsp_instancemanager = org.apache.jasper.runtime.InstanceManagerFactory.getInstanceManager(getServletConfig());
}
}
}
return _jsp_instancemanager;
}
public void _jspInit() {
}
public void _jspDestroy() {
}
public void _jspService(final javax.servlet.http.HttpServletRequest request, final javax.servlet.http.HttpServletResponse response)
throws java.io.IOException, javax.servlet.ServletException {
final java.lang.String _jspx_method = request.getMethod();
if (!"GET".equals(_jspx_method) && !"POST".equals(_jspx_method) && !"HEAD".equals(_jspx_method) && !javax.servlet.DispatcherType.ERROR.equals(request.getDispatcherType())) {
response.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, "JSPs only permit GET POST or HEAD");
return;
}
final javax.servlet.jsp.PageContext pageContext;
javax.servlet.http.HttpSession session = null;
final javax.servlet.ServletContext application;
final javax.servlet.ServletConfig config;
javax.servlet.jsp.JspWriter out = null;
final java.lang.Object page = this;
javax.servlet.jsp.JspWriter _jspx_out = null;
javax.servlet.jsp.PageContext _jspx_page_context = null;
try {
response.setContentType("text/html; charset=UTF-8");
pageContext = _jspxFactory.getPageContext(this, request, response,
null, true, 8192, true);
_jspx_page_context = pageContext;
application = pageContext.getServletContext();
config = pageContext.getServletConfig();
session = pageContext.getSession();
out = pageContext.getOut();
_jspx_out = out;
out.write("\n");
out.write("\n");
out.write("\n");
out.write("\n");
out.write("\n");
out.write("JSP compiled \n");
out.write("\n");
out.write("Hello World\n");
out.write("\n");
out.write("\n");
} catch (java.lang.Throwable t) {
if (!(t instanceof javax.servlet.jsp.SkipPageException)){
out = _jspx_out;
if (out != null && out.getBufferSize() != 0)
try {
if (response.isCommitted()) {
out.flush();
} else {
out.clearBuffer();
}
} catch (java.io.IOException e) {}
if (_jspx_page_context != null) _jspx_page_context.handlePageException(t);
else throw new ServletException(t);
}
} finally {
_jspxFactory.releasePageContext(_jspx_page_context);
}
}
}
如果你想看到你自己部署的Web服务器上的JSP转换出来的文件,可以到Tomcat的根目录下work/Catalina/应用名/org/apache/jsp/
下找到转换后的java源代码(请确保加载了该JSP页面,可以浏览器访问生成一下JSP)
所以我们就知道
JSP -> 转换为Java源码.java文件 -> 编译为Java编译后的.class文件
观察生成的源码我们知道页面主体被转换成一个_jspService
方法,这个方法在HttpJspPage里定义,并且通过HttpJspBase的service
方法实现调用。
Servlet容器会将及格对象给它运行的Servlet,例如HttpServletRequest和HttpServletResponse,并且在init
方法中中获得ServletConfig。
在JSP中可以通过使用九大隐式对象来获得这些对象,如下表所示:
对象 | 类型 |
---|---|
request | javax.servlet.HttpServletRequest |
response | javax.servlet.HttpServletResponse |
out | javax.servlet.jsp.JspWriter |
session | javax.servlet.http.HttpSession |
application | javax.servlet.ServletContext |
config | javax.servlet.ServletConfig |
pageContext | java.servlet.jsp.PageContext |
page | javax.servlet.jsp.HttpJspPage |
exception | java.lang.Throwable |
在编译后的servlet类_jspService
方法,就可以看到
final javax.servlet.jsp.PageContext pageContext;
javax.servlet.http.HttpSession session = null;
final javax.servlet.ServletContext application;
final javax.servlet.ServletConfig config;
javax.servlet.jsp.JspWriter out = null;
final java.lang.Object page = this;
javax.servlet.jsp.JspWriter _jspx_out = null;
javax.servlet.jsp.PageContext _jspx_page_context = null;
....
....
pageContext = _jspxFactory.getPageContext(this, request, response,
null, true, 8192, true);
_jspx_page_context = pageContext;
application = pageContext.getServletContext();
config = pageContext.getServletConfig();
session = pageContext.getSession();
out = pageContext.getOut();
_jspx_out = out;
比如在JSP代码中可以这样使用这些对象
<% String userName = request.getParameter("userName"); %>
pageContext是指为页面创建的javax.servlet.jsp.PageContext。它提供了一些方法来获得和Servlet有关的对象。比如getRequest
, getResponse
, getServletContext
, getServletConfig
以及 getSession
。这些没啥用,因为都是可以通过隐式对象访问到。
pageContext提供另外一些重要方法可以用来存取属性,getAttribute
和 setAttribute
方法等。
属性可以保存在一下四个范围内:page
,request
,session
,application
。其中page的范围最窄,仅能在当前页面能使用。还有请不要把隐式对象page当做是page域,而是指的隐式对象pageContext。request域指的是当前的ServletRequest,session域指的是HttpSession,application域指的是ServletCotext,简单的说就是你能调用getAttribute
和 setAttribute
的对象。
什么是当前页面范围呢,看_jspService
方法中
finally {
_jspxFactory.releasePageContext(_jspx_page_context);
}
最后释放了pageContext资源。而我们的request 可能会因为请求转发而继续存在,session本身就是记录用户信息的也能在一定时间内存在。application则是整个应用存在时间内有效。
<%-- comment --%>
jsp注释不会被发送到浏览器,不能嵌套
指令是第一种JSP元素,其指示JSP转换器应该如何将某个JSP页面转换成Servlet命令。JSP2.2中定义了几个指令,最重要的是这两个:page 和 include。
利用page指令可以就当前JSP页面的某些方面对JSP转换器提出指示。例如告诉JSP转换器隐式对象out应该有多大容量的缓存区,使用哪种内容类型。要导入那些Java类型。
语法如下:
<%@ page attribute1="value1" attribute2="value2"%>
page指令属性如下:
flush
方法page指令可以出现在任何地方,只有包含contentType和pageEncoding属性时才需要放在模版数据前。
include指令可以将一个文件的内容放到当前JSP页面中。
语法如下:
<%@ include file="url"%>
include方式有两种,一种是
,该种称为动态include,动态include主要是对动态页面的引入,它总是会检查所引入的页面的变化,如果所包含的资源在请求间发生变化,则下一次请求包含动作的jsp时,将包含资源的新内容。
另外一种是<%@ include file="url"%>
,include指令在转换时一次性地将内容复制到jsp中,如果所包含的资源发生变化,则使用include指令的jsp将不能反应出新的内容,除非重新编译该jsp。
动态的引入时需要频繁的变化和页面信息的更新和交互,要占用大量的资源开销。降低页面的访问速度。如果在没必要动态引入的情况下,不要使用动态include。
之前说过JSP中默认允许的脚本为java,脚本元素将Java代码合并成一个JSP页面。
脚本元素有三种:
Java代码块,以<%
开始,%>
结束。
<%
List week = new ArrayList<>();
week.add("Monday");
week.add("Tuesday");
week.add("Wednesday");
week.add("Thursday");
week.add("Friday");
week.add("Saturday");
week.add("Sunday");
%>
Hello World
<%
out.print("A weeks has:" + "
");
for(int i = 0; i < 7; i++) {
out.print(week.get(i) + "
");
}
%>
然后经过转换后生成的代码如下所示如下,上述页面中前面一个定义的Scriptlet变量后面是可见的。
List week = new ArrayList<>();
week.add("Monday");
week.add("Tuesday");
week.add("Wednesday");
week.add("Thursday");
week.add("Friday");
week.add("Saturday");
week.add("Sunday");
//html template start
out.write("\n");
out.write("Hello World
\n");
//html template end
out.print("A weeks has:" + "
");
for(int i = 0; i < 7; i++) {
out.print(week.get(i) + "
");
}
表达式的运算结果会被填入隐式对象out的print方法中,以<%=
开始, %>
结束。
如下
Today is <%=java.util.Calendar.getInstance().getTime()%>
表达式后 不需要分号等同于下面这个Scriptlet:
Today is
<%
out.print(java.util.Calendar.getInstance().getTime());
%>
可以声明在JSP页面中能够使用的变量和方法,以<%!
开始,%>
结束。
比如
<%!
List week = new ArrayList<>();
%>
<%
week.add("Monday");
week.add("Tuesday");
week.add("Wednesday");
week.add("Thursday");
week.add("Friday");
week.add("Saturday");
week.add("Sunday");
%>
Hello World
<%
showWeek(out);
%>
<%!
public void showWeek(JspWriter out) throws IOException{
out.print("A weeks has:" + "
");
for(int i = 0; i < 7; i++) {
out.print(week.get(i) + "
");
}
}
%>
还是和上个例子差不多,然后经过转换后生成的代码如下所示如下,可以发现在声明区域里的变量和方法都被统一移到生成代码类的前端定义了。声明可以在JSP任何位置定义,一个页面可以有多个声明。
public final class index_jsp extends org.apache.jasper.runtime.HttpJspBase implements org.apache.jasper.runtime.JspSourceDependent, org.apache.jasper.runtime.JspSourceImports {
List week = new ArrayList<>();
public void showWeek(JspWriter out) throws IOException{
out.print("A weeks has:" + "
");
for(int i = 0; i < 7; i++) {
out.print(week.get(i) + "
");
}
}
....
....
事实上现在开发很少用到脚本元素,大多是使用EL表达式和Tag,因为这和HTML语法类似,便于管理。
类似于Tag,动作提供以标签样式执行默写操作。类似下面的格式:
创建一个与某个Java类相关的变量,将表现逻辑和业务逻辑分隔开。事实上很少有了,因为有了强大的EL表达式。
<html> <head> head> <body> <jsp:useBean id="today" class="java.util.Date" /> <%=today%> body html
之前讲过动态的包含另外一个资源,可以是JSP页面,Servlet或者是静态页面。
forward将当前页面跳转到另外一个资源,该跳转是服务器这边的跳转。这里要说到另外一个概念就是redirect客户端方面的跳转。
forward后,服务器会将请求派遣至对应的servlet,实际上客户端那边根本不知道进行了跳转,地址栏不发生改变。
而redirect后,服务器返回30X状态码,告诉浏览器或者客户端,你请求的资源已经移到这个新的位置,所以客户端会跳转到这个新的位置去,地址栏发生改变。