JSP学习——JSP原理
1.什么是JSP?
JSP全名为Java Server Pages,中文名叫java服务器页面,其根本是一个简化的Servlet设计,它 [1] 是由Sun Microsystems公司倡导、许多公司参与一起建立的一种动态网页技术标准。
2.JSP相对与HTML的特点
JSP的语法与HTML相似,但是不同于HTML的是,JSP中可以写Java代码,HTML只能为用户静态数据,而正因为JSP中可以写Java代码,所以JSP可以动态的获取数据。
3.JSP的原理
我们知道浏览器想服务器发出一个请求,请求服务器资源的时候,其实都是在访问Servlet。所以当访问JSP页面的时候其实就是在访问Servlet,服务器在执行JSP的时候会先将JSP翻译成Servlet,然后再进行访问。说白了JSP其实就是Servlet的封装。也可以说JSP就是Servlet。
我们想要了解JSP就必须去看下JSP的源码,在C:\Users>IntelliJIdea2018.3>{用户}>system>tomcat>Web项目
当我们运行项目时,工作目录会出现下面两个文件,一个是我们默认的主页面index.jsp的java文件,一个是class文件。所以我们知道了原来JSP就是一个java类:
我们打开index.java源码分析:
源码:
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<java.lang.String,java.lang.Long> _jspx_dependants;
private static final java.util.Set<java.lang.String> _jspx_imports_packages;
private static final java.util.Set<java.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<java.lang.String,java.lang.Long> getDependants() {
return _jspx_dependants;
}
public java.util.Set<java.lang.String> getPackageImports() {
return _jspx_imports_packages;
}
public java.util.Set<java.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 {
if (!javax.servlet.DispatcherType.ERROR.equals(request.getDispatcherType())) {
final java.lang.String _jspx_method = request.getMethod();
if ("OPTIONS".equals(_jspx_method)) {
response.setHeader("Allow","GET, HEAD, POST, OPTIONS");
return;
}
if (!"GET".equals(_jspx_method) && !"POST".equals(_jspx_method) && !"HEAD".equals(_jspx_method)) {
response.setHeader("Allow","GET, HEAD, POST, OPTIONS");
response.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, "JSP 只允许 GET、POST 或 HEAD。Jasper 还允许 OPTIONS");
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("<html>\n");
out.write("<head>\n");
out.write(" <title>注册</title>\n");
out.write("</head>\n");
out.write("<body>\n");
out.write("\n");
out.write("\n");
out.write("<h1>注册</h1>\n");
out.write("\n");
out.write("<form action=\"");
out.write((java.lang.String) org.apache.jasper.runtime.PageContextImpl.proprietaryEvaluate("${pageContext.request.contextPath}", java.lang.String.class, (javax.servlet.jsp.PageContext)_jspx_page_context, null));
out.write("/q4.do\" method=\"post\">\n");
out.write(" 用户名:<input type=\"text\" name=\"username\"><br>\n");
out.write(" 密码:<input type=\"password\" name=\"password\"><br>\n");
out.write(" <input type=\"submit\">\n");
out.write(" <input type=\"reset\"><br>\n");
out.write("</form>\n");
out.write("</body>\n");
out.write("</html>");
} 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);
}
}
}
我们发现源码中有很多输出的东西,输出的是JSP的主页面index.jsp的文件内容:
index.jsp文件
一看,哎呀,原来jsp文件是这样来的!没错,一开始我们就说JSP就是一种Servlet,只是用流的形式给他展示成标签的形式,然后称之为JSP
然后我们再分析分析其中的方法,发现:源码中有初始化,销毁,和主要的service方法
初始化和销毁这两个方法见名知意,那么这个主要方法是干啥的?这个主要方法当然是解析JSP页面的
然后我们又看见这个源码中有一大堆属性:
这些属性又是干嘛的呢?
这八个对象就是JSP的内置对象,我们在Servlet中大部分都见过了,无非是换汤不换药,就是Servlet的东西。
原理这东西知道了更好,不知道也没啥太大的关系,只要知道JSP是Servlet就可以了,这就是JSP可以是实现动态WEB的原理,也是为什么JSP中可以写Java代码的原因。