百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

Java Web轻松学41 - JSTL初步使用

yuyutoo 2024-10-12 01:05 11 浏览 0 评论

本系列文章旨在记录和总结自己在Java Web开发之路上的知识点、经验、问题和思考,希望能帮助更多(Java)码农和想成为(Java)码农的人。

目录

  1. 介绍
  2. JSTL规范下载
  3. JSTL包含哪些库
  4. JSTL如何使用
  5. 租房网应用中使用JSTL
  6. JSTL相关的JAR包
  7. EL表达式访问列表的长度
  8. JSTL的forEach标签
  9. 剩下的页面改造
  10. 总结

介绍

上篇文章我们使用JSP技术对租房网平台进行了改造,也提到下面这样的代码有点奇葩:

<%for (House house : mockHouses) { 
	System.out.println(house); %>
	<li><h2><a href="house-details.jsp?userName=${param.userName}&houseId=<%=house.getId() %>"><%=house.getName() %></a></h2></li>
<%} %>

像这样的代码我们可以使用JSTL技术来解决。当然,JSTL可不仅仅只有这点功能,你还可以定义自己的标签。

这篇文章里,我提到过JSTL是JSP相关的技术,从它的名字全称(JSP Standard Tag Library,即JSP标准标签库)就可以看出来。

在本篇文章里,我们就尝试初步使用JSTL来进一步改造我们的租房网应用。

JSTL规范下载

既然称之为库,那么它包含哪些库呢?

我们可以把JSTL规范下载下来看一下,当然你也可以找本相关书籍,或者直接在网上搜索一下。

JSTL规范的下载类似Servlet规范的下载(可以参考这篇文章),不过我们在JCP官网(https://jcp.org/en/home/index)中搜索的关键字就变成Tag或Tag Library了。

我们在搜索结果中就可以看到:

点击 Download page 链接可以看到:

然后继续点击底部的 Maintenance Review 2 of JSR 52 链接,跳转到:

不过,再次点击DOWNLOAD按钮时,我这边出现无法访问此网站的错误。所以我转而点击底部的 JSR-000052 A Standard Tag Library for JavaServer Pages Detail Page 链接:

从这里我们可以看到JSTL规范的各个阶段,我们选择 Maintenance Release 2 这个阶段,点击它右边的 Download page 链接:

然后再点击红色箭头所指的链接,跳转到真正的下载页面:

后续操作就跟Servlet规范的下载类似了。

JSTL包含哪些库

现在,我们可以打开JSTL规范,可以看到如下描述:

事实上,JSTL应该就只是一个库,但它根据不同功能而划分成了多个库:

  • 核心:变量支持、流控制、URL管理等等,它的命名空间URI是:http://java.sun.com/jsp/jstl/core,标签前缀通常使用:c
  • XML处理:它的命名空间URI是:http://java.sun.com/jsp/jstl/xml,标签前缀通常使用:x
  • 国际化:语言区域、消息格式化、数字和日期格式化等,它的命名空间URI是:http://java.sun.com/jsp/jstl/fmt,标签前缀通常使用:fmt
  • 数据库访问(SQL):它的命名空间URI是:http://java.sun.com/jsp/jstl/sql,标签前缀通常使用:sql
  • 函数:集合长度、字符串操作等,它的命名空间URI是:http://java.sun.com/jsp/jstl/functions,标签前缀通常使用:fn

JSTL如何使用

JSTL的终极目标是简化JSP页面的开发,所以,它应该是在JSP页面中使用。

既然是标签,那它的使用是否跟HTML标签、XML标签类似呢?答案是肯定的。

不过,JSTL标签的使用与XML标签使用时声明命名空间类似,也需要告诉Servlet/JSP容器该JSP页面需要引入某个库(即上述的核心、XML处理、国际化、数据库访问、函数等等,以及以后自定义的标签)。

在JSP页面中是使用一个JSP指令(即taglib指令,之前我们用过page指令)来声明的:

<%@ taglib uri="uri" prefix="prefix" %>

举个例子,假设我们要使用JSTL的核心库,则应该在JSP页面的开头处这样声明:

<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>

然后,就可以使用核心库的标签,比如out标签:

<c:out value="value" [escapeXml="{true|false}"] [default="defaultValue"]/>
<c:out value="value" [escapeXml="{true|false}"]>
 default value
</c:out>

注意:在标签的语法中,[]表示可选的属性。如果值带下划线,则表示为默认值。

out标签有两种形式,有属性和属性值,也可能有标签内容,跟HTML标签和XML标签类似。

租房网应用中使用JSTL

我们就拿租房网应用中的房源列表页面houses.jsp来使用JSTL改造,因为这里涉及列表数据的展示。

列表数据是很常见的,我们经常可以看到包含列表数据的页面,比如订单列表、商品列表等等。

houses.jsp原来的代码是这样的:

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ include file="include.jsp"%>
<%
	List<House> mockHouses = (List<House>) request.getAttribute("mockHouses");
	System.out.println(mockHouses);
%>
<h6>共找到你感兴趣的房源 <%=mockHouses.size() %> 条</h6>
<ul>
<%for (House house : mockHouses) { 
	System.out.println(house); %>
	<li><h2><a href="house-details.jsp?userName=${param.userName}&houseId=<%=house.getId() %>"><%=house.getName() %></a></h2></li>
<%} %>
</ul>
</body>
</html>

首先要引入JSTL中的核心库(因为我们后面要用到的forEach标签是属于核心库的):

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ include file="include.jsp"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>

遗憾的是,我们添加这句声明之后,Eclipse就报错了:

提示是说找不到这个标签库的描述符。那就根据这个提示在网上搜索答案呗。

答案就是原来我们还需要在我们的工程结构里配置JSTL相关的JAR包,而Tomcat这个Servlet/JSP容器本身不提供这些JSTL相关的JAR包,我们需要单独下载。

JSTL相关的JAR包

那么在哪里下载呢?既然我们是使用Tomcat这个Servlet/JSP容器,那么我们就可以到它的官网(http://tomcat.apache.org/)上看看:

我们可以看到左侧导航栏中有个Taglibs链接,我们点进去看看:

还真就是JSTL相关JAR包的下载页面,点击Download链接进去,可以找到真正的下载链接:

我们可以看下Binary README文件,里面有该JAR包版本支持的Tomcat和JSP等版本信息,如果我们使用的Tomcat是最新版本(9.0.x),那直接下载下面四个JAR包即可,否则需要点进Archives页面下载历史版本。

好,我们把四个JAR包下载下来之后,就可以添加到我们的租房网应用的工程里面了,可以参考这篇文章

实际上,直接把这四个JAR包拷贝到WebContent/WEB-INF/lib节点下是最快的。

现在,我们可以看到上述的Eclipse报错提示就消失了。

EL表达式访问列表的长度

首先,我们可以使用EL表达式来访问列表的长度:

<h6>共找到你感兴趣的房源 ${mockHouses.size()} 条</h6>

据说低版本的JSP容器不可以这样直接用EL表达式来访问列表的长度,而应该使用JSTL的函数库中的length方法:

${fn:length(list) }

当然,首先需要引入JSTL的函数库。读者朋友可以自行尝试一下。

JSTL的forEach标签

我们可以通过JSTL规范查看一下这个forEach标签的用法:

当然,可能只看这个语法格式还是不太明白如何使用,那么可以继续看规范中的描述,或者直接网上搜索即可。

我们需要改造的代码是这一部分:

<%for (House house : mockHouses) { 
	System.out.println(house); %>
	<li><h2><a href="house-details.jsp?userName=${param.userName}&houseId=<%=house.getId() %>"><%=house.getName() %></a></h2></li>
<%} %>

我们可以先把它注释掉(在Eclipse中可以选中这一段,然后键入Ctrl + Shift + /),然后敲入开始标签的左尖括号 < ,于是Eclipse会出现智能提示:

嗯,这个功能还是挺好用的,提高开发效率。

闲话不多说了,直接上代码:

<c:forEach var="house" items="${mockHouses}">
	<li><h2><a href="house-details.jsp?userName=${param.userName}&houseId=${house.id}">${house.name}</a></h2></li>
</c:forEach>

看似跟之前变化不大,但至少没那么奇葩了,风格也与HTML很一致。

forEach标签遍历列表数据的基本属性是:

  • var:相当于定义一个变量来表示指向列表中的某一项;
  • items:指定需要遍历哪个列表,需要注意的是其值又是使用EL表达式来访问页面/请求/会话/应用中的某个属性数据(即使用setAttribute()添加的对象)

forEach标签的内容就可以是普通的HTML标签了,然后HTML标签里我们就可以使用EL表达式来访问列表中每一项数据。

实际上,EL表达式直接访问对象的属性,而非调用对象的方法,比如:

${house.id}

注意,param是EL的隐式对象,它是由Servlet/JSP容器创建并传进来的,它可以直接访问请求所携带的参数。

所以,我们实际上可以把原来的这一部分:

<%
	List<House> mockHouses = (List<House>) request.getAttribute("mockHouses");
	System.out.println(mockHouses);
%>

删除掉。

现在houses.jsp的代码就变成这样了:

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ include file="include.jsp"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
<h6>共找到你感兴趣的房源 ${mockHouses.size()} 条</h6>
<ul>
<c:forEach var="house" items="${mockHouses}">
	<li><h2><a href="house-details.jsp?userName=${param.userName}&houseId=${house.id}">${house.name}</a></h2></li>
</c:forEach>
</ul>
</body>
</html>

是不是清晰明朗了许多?

当然,JSP页面的开发者必须知道传进这个页面的到底有哪些数据对象!

没错,这就相当于把数据的展示(视图层)给分离开来,你需要与后端(控制器层和模型层)约定/设计好每一个JSP页面都有哪些数据对象。

然后,JSP页面的开发者和后端(控制器层和模型层)的开发者就可以各自独立去开发了。

剩下的页面改造

剩下的house-details.jsp和house-form.jsp该如何改造呢?大家可以先思考一下。

提示:JSP页面应该只关心取数据展示,而不应该关心如何查找到某个数据这种逻辑,应该把这种逻辑放到后端(控制器层和模型层)。

我的改造是这样的,把原来这两个页面的如何查找到某个数据这种逻辑放到Filter中,前后端约定好这两个页面中存在target这个House对象,于是doFilter()方法中在将请求交给下个节点之前应该挂载上target这个House对象:

		if (userName == null || userName.isEmpty()) {
			System.out.println("invalid user!");
			httpServletResponse.sendRedirect("login.html");
		} else {
			String houseId = httpServletRequest.getParameter("houseId");
			if (houseId != null && !houseId.trim().isEmpty()) {
				House target = findHouseById(houseId);//找不到怎么办?
				httpServletRequest.setAttribute("target", target);
			}
			chain.doFilter(request, response);
		}

另外,应该把如何查找某个房源的逻辑封装起来:

	private House findHouseById(String houseId) {
		for (House house : mockHouses) {
			if (houseId.equals(house.getId())) {
				return house;
			}
		}
		return null;
	}

然后,这两个页面就极其简单了,house-details.jsp是这样的:

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ include file="include.jsp"%>
<h2>${target.name}<a href="house-form.jsp?userName=${param.userName}&houseId=${target.id}">编辑</a></h2>
<h3>${target.detail}</h3>
<h4><a href="houses.jsp?userName=${param.userName}">回到列表</a></h4>
</body>
</html>

house-form.jsp是这样的:

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ include file="include.jsp"%>
<form action="house-form.servlet" method="post">
<input type="hidden" name="userName" value="${param.userName}"/>
<input type="hidden" name="houseId" value="${target.id}"/>
<label for="house_name">房源名字:</label><input type="text" id="house_name" name="houseName" value="${target.name}" />
<label for="house_detail">房源详细信息:</label><input type="text" id="house_detail" name="houseDetail" value="${target.detail}" />
<input type="submit" value="提交" />
</form>
</body>
</html>

可以看到,这两个页面只是取数据(target这个House对象)展示而已。

总结

  • JSTL的终极目标是简化JSP开发;
  • JSTL的使用需要先声明某个库;
  • JSTL的标签使用类似HTML标签和XML标签,语法规则可以查看JSTL规范;
  • JSP页面使用JSTL和EL表达式基本可以实现大部分功能;
  • EL表达式能够直接访问页面/请求/会话/应用这几个对象中挂载(即setAttribute()方法)的数据对象;
  • EL表达式使用 ${ }
  • EL中存在隐式对象,以后再介绍;
  • 列表数据很常见很重要;
  • 列表数据采用JSTL的forEach标签来遍历(实际上也可以采用EL表达式,比如${mockHouses[i].id});
  • 前后端开发的分离:前端只关心取数据如何展示;后端关心如何取数据并挂载到页面/请求/会话/应用这几个对象中;
  • 前后端需约定/设计好数据;

相关推荐

当 Linux 根分区 (/) 已满时如何释放空间?

根分区(/)是Linux文件系统的核心,包含操作系统核心文件、配置文件、日志文件、缓存和用户数据等。当根分区满载时,系统可能出现无法写入新文件、应用程序崩溃甚至无法启动的情况。常见原因包括:...

玩转 Linux 之:磁盘分区、挂载知多少?

今天来聊聊linux下磁盘分区、挂载的问题,篇幅所限,不会聊的太底层,纯当科普!!1、Linux分区简介1.1主分区vs扩展分区硬盘分区表中最多能存储四个分区,但我们实际使用时一般只分为两...

Linux 文件搜索神器 find 实战详解,建议收藏

在Linux系统使用中,作为一个管理员,我希望能查找系统中所有的大小超过200M文件,查看近7天系统中哪些文件被修改过,找出所有子目录中的可执行文件,这些任务需求...

Linux 操作系统磁盘操作(linux 磁盘命令)

一、文档介绍本文档描述Linux操作系统下多种场景下的磁盘操作情况。二、名词解释...

Win10新版19603推送:一键清理磁盘空间、首次集成Linux文件管理器

继上周四的Build19592后,微软今晨面向快速通道的Insider会员推送Windows10新预览版,操作系统版本号Build19603。除了一些常规修复,本次更新还带了不少新功能,一起来了...

Android 16允许Linux终端使用手机全部存储空间

IT之家4月20日消息,谷歌Pixel手机正朝着成为强大便携式计算设备的目标迈进。2025年3月的更新中,Linux终端应用的推出为这一转变奠定了重要基础。该应用允许兼容的安卓设备...

Linux 系统管理大容量磁盘(2TB+)操作指南

对于容量超过2TB的磁盘,传统MBR分区表的32位寻址机制存在限制(最大支持2.2TB)。需采用GPT(GUIDPartitionTable)分区方案,其支持64位寻址,理论上限为9.4ZB(9....

Linux 服务器上查看磁盘类型的方法

方法1:使用lsblk命令lsblk输出说明:TYPE列显示设备类型,如disk(物理磁盘)、part(分区)、rom(只读存储)等。...

ESXI7虚机上的Ubuntu Linux 22.04 LVM空间扩容操作记录

本人在实际的使用中经常遇到Vmware上安装的Linux虚机的LVM扩容情况,最终实现lv的扩容,大多数情况因为虚机都是有备用或者可停机的情况,一般情况下通过添加一块物理盘再加入vg,然后扩容lv来实...

5.4K Star很容易!Windows读取Linux磁盘格式工具

[开源日记],分享10k+Star的优质开源项目...

Linux 文件系统监控:用脚本自动化磁盘空间管理

在Linux系统中,文件系统监控是一项非常重要的任务,它可以帮助我们及时发现磁盘空间不足的问题,避免因磁盘满而导致的系统服务不可用。通过编写脚本自动化磁盘空间管理,我们可以更加高效地处理这一问题。下面...

Linux磁盘管理LVM实战(linux实验磁盘管理)

LVM(逻辑卷管理器,LogicalVolumeManager)是一种在Linux系统中用于灵活管理磁盘空间的技术,通过将物理磁盘抽象为逻辑卷,实现动态调整存储容量、跨磁盘扩展等功能。本章节...

Linux查看文件大小:`ls`和`du`为何结果不同?一文讲透原理!

Linux查看文件大小:ls和du为何结果不同?一文讲透原理!在Linux运维中,查看文件大小是日常高频操作。但你是否遇到过以下困惑?...

使用 df 命令检查服务器磁盘满了,但用 du 命令发现实际小于磁盘容量

在Linux系统中,管理员或开发者经常会遇到一个令人困惑的问题:使用...

Linux磁盘爆满紧急救援指南:5步清理释放50GB+小白也能轻松搞定

“服务器卡死?网站崩溃?当Linux系统弹出‘Nospaceleft’的红色警报,别慌!本文手把手教你从‘删库到跑路’进阶为‘磁盘清理大师’,5个关键步骤+30条救命命令,快速释放磁盘空间,拯救你...

取消回复欢迎 发表评论: