JSP与EL表达式与JSTL标签库

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语法

  1. 在jsp文件中直接编写文字会被翻译到servlet类的service方法的out.write("翻译到这里"),直接翻译到双引号里,被java程序当做普通字符串打印输出到浏览器。
    (在JSP中编写的HTML CSS JS代码,这些代码对于JSP来说只是一个普通的字符串。但是JSP把这个普通的字符串一旦输出到浏览器,浏览器就会对HTML CSS JS进行解释执行。展现一个效果.)

  2. 在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方法当中了
  1. 在JSP中如何编写JSP的专业注释

    • <%--JSP的专业注释,不会被翻译到java源代码当中。--%>
  2. JSP语法总结

    • JSP中直接编写普通字符串
      • 翻译到service方法的out.write("这里")
    • <%%>
      • 翻译到service方法体内部,里面是一条一条的java语句。
    • <%! %>
      • 翻译到service方法之外。
    • <%= %>
      • 翻译到service方法体内部,翻译为:out.print();
    • <%@page contentType="text/html;charset=UTF-8"%>
      • page指令,通过contentType属性用来设置响应的内容类型。

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表达式

  1. Expression Language(表达式语言)
  2. EL表达式可以代替JSP中的java代码,让JSP文件中的程序看起来更加整洁,美观。
  3. JSP中夹杂着各种java代码,例如<% java代码 %>、<%=%>等,导致JSP文件很混乱,不好看,不好维护。所以才有了后期的EL表达式。
  4. EL表达式可以算是JSP语法的一部分。EL表达式归属于JSP。
  5. EL表达式出现在JSP中主要是:
    • 从某个作用域中取数据,然后将其转换成字符串,然后将其输出到浏览器。这就是EL表达式的功效。三大功效:
    • 第一功效:从某个域中取数据。
    • 四个域:
      • pageContext
      • request
      • session
      • application
    • 第二功效:将取出的数据转成字符串。
      • 如果是一个java对象,也会自动调用java对象的toString方法将其转换成字符串。
    • 第三功效:将字符串输出到浏览器。
      • 和这个一样:<%= %>,将其输出到浏览器。
  6. 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表达式中内置对象

  1. pageContext
  2. param
  3. paramValues
  4. initParam
  5. 其他(不是重点)

四个重要隐式对象

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的步骤

  1. 引入JSTL对应的jar包

    • 在WEB-INF下面新建lib目录 然后把对应的jar包 复制过去
  2. 在JSP中引入要使用的标签库。

    • 使用taglib指令引入标签库

在JSTL中有很多标签 ,我们只需要重点掌握核心标签库

<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
     prefix="随便起名字" 
  1. 在需要使用标签的位置使用即可

表面使用的是标签,底层实际上还是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>
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,076评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,658评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,732评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,493评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,591评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,598评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,601评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,348评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,797评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,114评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,278评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,953评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,585评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,202评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,180评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,139评论 2 352

推荐阅读更多精彩内容