JSP
JSP程序
我的第一个JSP程序:
- 在WEB-INF目录之外创建一个index.jsp文件,然后这个文件中没有任何内容。
- 将上面的项目部署之后,启动服务器,打开浏览器,访问以下地址:
- http://localhost:8080/jsp/index.jsp 展现在大家面前的是一个空白。
过程:
- 实际上访问以上的这个:index.jsp,底层执行的是:index_jsp.class 这个java程序。
- 这个index.jsp会被tomcat翻译生成index_jsp.java文件,然后tomcat服务器又会将index_jsp.java编译生成index_jsp.class文件
- 访问index.jsp,实际上执行的是index_jsp.class中的方法。
JSP生命周期
JSP实际上就是一个Servlet
- index.jsp访问的时候,会自动翻译生成index_jsp.java,会自动编译生成index_jsp.class,那么index_jsp 这就是一个类。
- index_jsp 类继承 HttpJspBase,而HttpJspBase类继承的是HttpServlet。所以index_jsp类就是一个Servlet类。
- jsp的生命周期和Servlet的生命周期完全相同。完全就是一个东西。没有任何区别。
- jsp和servlet一样,都是单例的。(假单例)
思考
jsp文件第一次访问的时候是比较慢的,为什么?
为什么大部分的运维人员在给客户演示项目的时候,为什么提前先把所有的jsp文件先访问一遍。
第一次比较麻烦:
- 要把jsp文件翻译生成java源文件
- java源文件要编译生成class字节码文件
- 然后通过class去创建servlet对象
- 然后调用servlet对象的init方法
- 最后调用servlet对象的service方法。
第二次就比较快了,为什么?
因为第二次直接调用单例servlet对象的service方法即可。
JSP定义
JSP是什么?
JSP是java程序。(JSP本质还是一个Servlet)
- JSP是:JavaServer Pages的缩写。(基于Java语言实现的服务器端的页面。)
- Servlet是JavaEE的13个子规范之一,那么JSP也是JavaEE的13个子规范之一。
- JSP是一套规范。所有的web容器/web服务器都是遵循这套规范的,都是按照这套规范进行的“翻译”
- 每一个web容器/web服务器都会内置一个JSP翻译引擎。
- 对JSP进行错误调试的时候,还是要直接打开JSP文件对应的java文件,检查java代码。
开发JSP的最高境界
眼前是JSP代码,但是脑袋中呈现的是java代码。
JSP语法
在jsp文件中直接编写文字会被翻译到servlet类的service方法的out.write("翻译到这里"),直接翻译到双引号里,被java程序当做普通字符串打印输出到浏览器。
(在JSP中编写的HTML CSS JS代码,这些代码对于JSP来说只是一个普通的字符串。但是JSP把这个普通的字符串一旦输出到浏览器,浏览器就会对HTML CSS JS进行解释执行。展现一个效果.)-
在JSP中编写java程序
<% java语句; %> //向浏览器上输出一个java变量。 <% String name = “jack”; out.write("name = " + name); %>
- 在这个符号当中编写的被视为java程序,被翻译到Servlet类的service方法内部。
- 在service方法当中编写的代码是有顺序的,方法体当中的代码要遵循自上而下的顺序依次逐行执行。
- service方法当中不能写静态代码块,不能写方法,不能定义成员变量
- 在同一个JSP当中 <%%> 这个符号可以出现多个。
- 注意:以上代码中的out是JSP的九大内置对象之一。可以直接拿来用。当然,必须只能在service方法内部使用。
如果输出的内容中含有“java代码”,这个时候可以使用以下语法格式:
<%= %>
<%= %> 这个符号会被翻译到哪里?最终翻译成什么?
- 翻译成了这个java代码:
out.print();
- 翻译到service方法当中了
-
在JSP中如何编写JSP的专业注释
- <%--JSP的专业注释,不会被翻译到java源代码当中。--%>
-
JSP语法总结
- JSP中直接编写普通字符串
- 翻译到service方法的out.write("这里")
- <%%>
- 翻译到service方法体内部,里面是一条一条的java语句。
- <%! %>
- 翻译到service方法之外。
- <%= %>
- 翻译到service方法体内部,翻译为:out.print();
- <%@page contentType="text/html;charset=UTF-8"%>
- page指令,通过contentType属性用来设置响应的内容类型。
- JSP中直接编写普通字符串
JSP指令
指令的作用:指导JSP的翻译引擎如何工作(指导当前的JSP翻译引擎如何翻译JSP文件。)
指令包括哪些呢?
- include指令:包含指令,在JSP中完成静态包含,很少用了。(这里不讲)
- taglib指令:引入标签库的指令。这个到JJSTL标签库的时候再学习。现在先不管。
- page指令:目前重点学习一个page指令。
指令的使用语法是什么?
<%@指令名 属性名=属性值 属性名=属性值 属性名=属性值....%>
关于page指令当中都有哪些常用的属性呢?
<%@page session="true|false" %>
true表示启用JSP的内置对象session,表示一定启动session对象。没有session对象会创建。
如果没有设置,默认值就是session="true"
session="false" 表示不启用内置对象session。当前JSP页面中无法使用内置对象session。
<%@page contentType="text/json" %>
contentType属性用来设置响应的内容类型
但同时也可以设置字符集。
<%@page contentType="text/json;charset=UTF-8" %>
<%@page import="java.util.List, java.util.Date, java.util.ArrayList" %>
<%@page import="java.util.*" %>
import语句,导包
<%@page errorPage="/error.jsp" %>
当前页面出现异常之后,跳转到error.jsp页面。
errorPage属性用来指定出错之后的跳转位置。
<%@page isErrorPage="true" %>
表示启用JSP九大内置对象之一:exception
默认值是false
JSP九大内置对象
- jakarta.servlet.jsp.PageContext pageContext 页面作用域
- jakarta.servlet.http.HttpServletRequest request 请求作用域
- jakarta.servlet.http.HttpSession session 会话作用域
- jakarta.servlet.ServletContext application 应用作用域
- pageContext < request < session < application
- 以上四个作用域都有:setAttribute、getAttribute、removeAttribute方法。
- 以上作用域的使用原则:尽可能使用小的域。
- java.lang.Throwable exception
- jakarta.servlet.ServletConfig config
- java.lang.Object page (其实是this,当前的servlet对象)
- jakarta.servlet.jsp.JspWriter out (负责输出)
- jakarta.servlet.http.HttpServletResponse response (负责响应)
EL表达式
什么是EL表达式
- Expression Language(表达式语言)
- EL表达式可以代替JSP中的java代码,让JSP文件中的程序看起来更加整洁,美观。
- JSP中夹杂着各种java代码,例如<% java代码 %>、<%=%>等,导致JSP文件很混乱,不好看,不好维护。所以才有了后期的EL表达式。
- EL表达式可以算是JSP语法的一部分。EL表达式归属于JSP。
- EL表达式出现在JSP中主要是:
- 从某个作用域中取数据,然后将其转换成字符串,然后将其输出到浏览器。这就是EL表达式的功效。三大功效:
- 第一功效:从某个域中取数据。
- 四个域:
- pageContext
- request
- session
- application
- 第二功效:将取出的数据转成字符串。
- 如果是一个java对象,也会自动调用java对象的toString方法将其转换成字符串。
- 第三功效:将字符串输出到浏览器。
- 和这个一样:<%= %>,将其输出到浏览器。
- EL表达式很好用,基本的语法格式:
- ${表达式}
EL表达式的使用
<%
// 创建User对象
User user = new User();
user.setUsername("jackson");
user.setPassword("1234");
user.setAge(50);
// 将User对象存储到某个域当中。一定要存,因为EL表达式只能从某个范围中取数据。
// 数据是必须存储到四大范围之一的。
request.setAttribute("userObj", user);
%>
<%--使用EL表达式取--%>
${这个位置写什么????这里写的一定是存储到域对象当中时的name}
要这样写:
${userObj}
等同于java代码:<%=request.getAttribute("userObj")%>
你不要这样写:${"userObj"}
面试题:
${abc} 和 ${"abc"}的区别是什么?
${abc}表示从某个域中取出数据,并且被取的这个数据的name是"abc",之前一定有这样的代码: 域.setAttribute("abc", 对象);
${"abc"} 表示直接将"abc"当做普通字符串输出到浏览器。不会从某个域中取数据了。
${userObj} 底层是怎么做的?从域中取数据,取出user对象,然后调用user对象的toString方法,转换成字符串,输出到浏览器。
<%--如果想输出对象的属性值,怎么办?--%>
${userObj.username} 使用这个语法的前提是:User对象有getUsername()方法。
${userObj.password} 使用这个语法的前提是:User对象有getPassword()方法。
${userObj.age} 使用这个语法的前提是:User对象有getAge()方法。
${userObj.email} 使用这个语法的前提是:User对象有getEmail()方法。
EL表达式中的. 这个语法,实际上调用了底层的getXxx()方法。
注意:如果没有对应的get方法,则出现异常。报500错误。
${userObj.addr222.zipcode}
以上EL表达式对应的java代码:
user.getAddr222().getZipcode()
EL表达式优先从小范围中读取数据。
pageContext < request < session < application
EL表达式中有四个隐含的隐式的范围:
- pageScope 对应的是 pageContext范围。
- requestScope 对应的是 request范围。
- sessionScope 对应的是 session范围。
- applicationScope 对应的是 application范围。
EL表达式对null进行了预处理。如果是null,则向浏览器输出一个空字符串。
EL表达式取数据的时候有两种形式:
- 第一种:. (大部分使用这种方式)
- 第二种:[ ] (如果存储到域的时候,这个name中含有特殊字符,可以使用 [ ])
- request.setAttribute("abc.def", "zhangsan");
- ${requestScope.abc.def} 这样是无法取值的。
- 应该这样:${requestScope["abc.def"]}
从Map集合中取数据
掌握使用EL表达式,怎么从Map集合中取数据:
${map.key}
从数组和list集合中取数据
掌握使用EL表达式,怎么从数组和List集合中取数据
- ${数组[0]
- ${数组[1]}
- ${list[0]}
其它
page指令当中,有一个属性,可以忽略EL表达式
<%@page contentType="text/html;charset=UTF-8" isELIgnored="true" %>
isELIgnored="true" 表示忽略EL表达式
isELIgnored="false" 表示不忽略EL表达式。(这是默认值)
isELIgnored="true" 这个是全局的控制。
可以使用反斜杠进行局部控制:\${username} 这样也可以忽略EL表达式。
EL表达式中内置对象
- pageContext
- param
- paramValues
- initParam
- 其他(不是重点)
四个重要隐式对象
pageContext
应用的根路径:${pageContext.request.contextPath}
param
用户在浏览器上提交数据:http://localhost:8080/EL/1.jsp?username=jack
用户名:<%=request.getParameter("username")%>
//使用param
用户名:${param.username}
paramValues
用户在浏览器上提交数据:http://localhost:8080/EL/1.jsp?aihao=smoke&aihao=drike&aihao=tangtou
爱好:${paramValues.aihao[0]}、${paramValues.aihao[1]}、${paramValues.aihao[2]}
initParam
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee https://jakarta.ee/xml/ns/jakartaee/web-app_5_0.xsd"
version="5.0">
<!--Servlet上下文初始化参数-->
<!--上下文初始化参数被封装到:ServletContext对象当中了-->
<context-param>
<param-name>pageSize</param-name>
<param-value>20</param-value>
</context-param>
</web-app>
<%
String a =application.getInitParameter("pageSize");
%>
每页显示的记录条数:<%=a%> //20
//使用initParam
每页显示的记录条数${initParam.pageSize} //20
EL表达式中的运算符
算数运算符
+只能做求和运算,不会进行字符串的拼接
两边不是数字的时候,一定会转成数字,转不成就报错:NumberFormatException
关系运算符
在EL表达式中 ==
调用了equals
方法
<%
Student stu1 = new Student("110",警察);
Student stu2 = new Student("110",警察);
request.setAttribute("stu1",stu1);
request.setAttribute("stu2",stu2);
%>
${stu1==stu2} //true
<%
Object o1 = new Object();
Object o2 = new Object();
request.setAttribute("o1",o1);
request.setAttribute("o2",o2);
%>
${o1==o2}// fasle
在EL表达式中 eq
调用了equals
方法
<%
Student stu1 = new Student("110",警察);
Student stu2 = new Student("110",警察);
request.setAttribute("stu1",stu1);
request.setAttribute("stu2",stu2);
%>
${stu1 eq stu2} //true
在EL表达式中 !=
调用了equals
方法
<%
Student stu1 = new Student("110",警察);
Student stu2 = new Student("110",警察);
request.setAttribute("stu1",stu1);
request.setAttribute("stu2",stu2);
%>
${stu1 != stu2} //false
条件运算符
? :
${empty param.username ? "用户名不能为空" : "欢迎访问!"}
empty运算符
判断是否为空 如果为空结果是true,如果不为空结果是false
${empty param.username}
JSTL标签库
什么是JSTL标签库
- Java Standard Tag Lib (Java标准的标签库)
- JSTL标签库通常结合EL表达式一起使用。目的是让JSP中的java代码消失。
- 标签是写在JSP当中的,但实际上最终还是要执行对应的java程序。(java程序在jar包当中。)
使用JSTL的步骤
-
引入JSTL对应的jar包
- 在WEB-INF下面新建lib目录 然后把对应的jar包 复制过去
-
在JSP中引入要使用的标签库。
- 使用taglib指令引入标签库
在JSTL中有很多标签 ,我们只需要重点掌握核心标签库
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
prefix="随便起名字"
- 在需要使用标签的位置使用即可
表面使用的是标签,底层实际上还是java
程序
JSTL标签原理
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
以上uri后面的路径实际上指向了一个×x×.tld文件。
tld文件实际上是一个xml配置文件。
在tld文件中描述了"标签"和"java类"之间的关系。
以上核心标签库对应的tld文件是:c.tld文件。它在哪里。
在jakarta.serv1et.jsp.jst1-2.0.0.jar里面META-INF目录下,有一个c.t1d文件。
配置文件:tld解析
<tag>
<description>对标签的描述</description>
<name>标签的名字</name>
<tag-class>标签对应java类</tag-class>
<body-content>JSP</body-content>标签体当中可以出现的内容,如果是JSp,
就表示标签体中可以出现符合JSp所有语法的代码。例如EL表达式。
<attribute>
<description>
对这个属性的描述
</description>
<name>var</name>属性名
<required>false</required>false表示该属性不是必须的
<rtexprvalue>false</rtexprvalue>false 不支持EL表达式
</attribute>
</tag>
常用的标签
JSTL中的核心标签库core当中有哪些常用的标签呢?
<c:if text="boolean类型,支持EL表达式"></c:if>
<c:forEach items="集合,支持EL表达式" var = "集合中的元素" varStatus="元素的状态对象">${元素状态对象:count}</c:forEach>
<c:forEach var="i" begin="1" end="10" step="2"> ${i}</c:forEach>
<c:choose>
<c:when test="${param.age < 18}">
青少年
</c:when>
<c:when test="${param.age < 35}">
青年
</c:when>
<c:when test="${param.age < 55}">
中年
</c:when>
<c:otherwise>
老年
</c:otherwise>
</c:choose>