JSP实现自定义标签【上】

news2025/1/8 5:58:25

目录

一、基础概念

1、标签语言的形式或结构

2、分类

二、自定义标签的开发及步骤

三、标签生命周期

1、返回值

四、案例

1、if

2、out


一、基础概念

JSP自定义标签是一种扩展JSP标记语言的方法。通过自定义标签,我们可以将自定义功能封装在一个独立的标签中,以供重复使用。

自定义标签通常由两个部分组成:标签处理器(Tag Handler)和标签库描述文件(Tag Library Descriptor)。标签处理器负责解析自定义标签,并在页面中生成相应的HTML输出;而标签库描述文件则提供了自定义标签的元数据信息,包括标签名称、属性、使用方式等。

对于标签处理器,通常有两种实现方式:标签文件和标签类。标签文件是一个以“.tag”结尾的文件,其中包含了标签处理器的Java代码。而标签类则是一个Java类,实现了javax.servlet.jsp.tagext.Tag接口或其子接口,重写了相应的方法来实现标签的处理逻辑。

标签库描述文件通常是一个XML文件,描述了所定义的自定义标签的属性、使用方法和标签处理器的类路径等信息。标签库描述文件必须被放置在WEB-INF目录下的“tld”子目录中,并在JSP页面的“<%@ taglib %>”指令中声明。

自定义标签是JSP技术的一个重要组成部分,可以帮助开发人员更加方便地构建JSP页面,提高代码复用性和可维护性。

1、标签语言的形式或结构

<开始标签 属性="属性值">标签体</结束标签>

代码示例:

<%@ page language="java" contentType="text/html; charset=UTF-8"
	pageEncoding="UTF-8"%>
<%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Demo01</title>
</head>
<body>
<!--
	 标签结构:
		开始标签
		标签体
		结束标签
	标签的分类:
		空标签
		控制标签
		数据标签
		UI标签:没有标签体也能在网页中输出内容
		
	通过点击c:if标签,可以跳转到c.tld文件中,而tld文件就是标签库的定义配置文件
	
 -->
	<c:if test="true">true</c:if>
	<c:if test="false">false</c:if>
	<c:set var="name" value="zs"></c:set>
	<c:out value="${name }"></c:out>
</body>
</html>

输出结果:

2、分类

空标签如:br、hr
控制标签如:if、foreach
数据标签如:out标签
UI标签如:input、table

二、自定义标签的开发及步骤

  1. 自定义标签库与tld文件相关
  2. 标签库中的标签与tld中的tag元素有关

步骤:

  1. 必须要编写助手类并继承(BodyTagSupport
  2. 编写标签库描述文件(tld):必须保存到WEB-INF目录或者其子目录
  3. JSP通过taglib 指令导入标签库

我们实现一个自己写的一个tld文件

<?xml version="1.0" encoding="UTF-8" ?>

<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd"
    version="2.0">
    
  <description>JSTL 1.1 core library</description>
  <display-name>JSTL core</display-name>
  <tlib-version>1.1</tlib-version>
  <!-- 一个名字 -->
  <short-name>t</short-name>
  <!-- 路径 -->
  <uri>http://jsp.tgq.cn</uri>

<tag>
	<!-- 标签库名 -->
    <name>demo1</name>
    <!-- 对应的助手类 -->
    <tag-class>com.tgq.jsp01.DemoTag01</tag-class>
    <!-- 代表JSP标签 -->
    <body-content>JSP</body-content>
    <!-- <attribute>
    自定义JSP标签的属性名称 
        <name>var</name>
        该属性是否必填
        <required>false</required>
        该属性值是否支持表达式
        <rtexprvalue>false</rtexprvalue>
    </attribute> -->
</tag>
</taglib>

助手类

package com.tgq.jsp01;

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;

/**
 * 助手类 必须继承bodytagSupport
 * 
 * @author tgq
 *
 */
public class DemoTag01 extends BodyTagSupport {

	// 重写doStartTag方法
	@Override
	public int doStartTag() throws JspException {
		System.out.println("======doStartTag========");
		 return super.doStartTag();
	}

	// 重写doAfterBody方法
	// 标签体
	@Override
	public int doAfterBody() throws JspException {
		System.out.println("======doAfterBody========");
		return super.doAfterBody();
		
	}

	@Override
	public int doEndTag() throws JspException {
		System.out.println("=======doEndTag=======");
	return super.doEndTag();

	}

}

JSP代码测试

<%@ page language="java" contentType="text/html; charset=UTF-8"
	pageEncoding="UTF-8"%>
<%@taglib uri="http://jsp.tgq.cn" prefix="t"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Demo03</title>
</head>
<body>
	<!-- 自定义标签开发及使用步骤 -->

	<t:demo1>zxcv</t:demo1>
</body>
</html>

输出结果:

三、标签生命周期

1、返回值

我们常用的几个返回值:

  1. SKIP_BODY:跳过主体
  2. EVAL_BODY_INCLUDE:计算标签主体内容并输出
  3. EVAL_BODY_AGAIN:再计算主体一次
  4. EVAL_PAGE:计算页面的后续部分
  5. SKIP_PAGE:跳过页面的后续部分

JSP代码:

<%@ page language="java" contentType="text/html; charset=UTF-8"
	pageEncoding="UTF-8"%>
<%@taglib uri="http://jsp.tgq.cn" prefix="t"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Demo03</title>
</head>
<body>
	<!-- 自定义标签开发及使用步骤 -->
	<!-- 
		jsp标签的生命周期
	
		1、有标签体的情况下,默认会调用助手类的doStartTag、doAfterBody、doEndTag
		2、如果将doStartTag返回值改为SKIP_BODY,doAfterBody不会调用执行(路线1)
		3、如果将doStartTag返回值改为EVAL_BODY_INCLUDE,那么就会执行(路线2)
		4、如果将doAfterBody返回值改为EVAL_BODY_AGAIN,会一直调用doAfterBody方法进入循环(路线3)
		
		 -->
	<t:demo1>zxcv</t:demo1>
	<!-- 如果不想显示这段只要将 doEndTag的返回值 改为SKIP_PAGE-->
	rtyu
</body>
</html>

输出代码我就不演示了我直接给大家看一下助手类的代码,自己进行测试

package com.tgq.jsp01;

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;

/**
 * 助手类 必须继承bodytagSupport
 * 
 * @author tgq
 *
 */
public class DemoTag01 extends BodyTagSupport {

	// 重写doStartTag方法
	@Override
	public int doStartTag() throws JspException {
		System.out.println("======doStartTag========");
		// 如果想标签体有东西有又不显示就使用:跳过这个标签体
//		return SKIP_BODY;
		return EVAL_BODY_INCLUDE;
		// return super.doStartTag();
	}

	// 重写doAfterBody方法
	// 标签体
	@Override
	public int doAfterBody() throws JspException {
		System.out.println("======doAfterBody========");
		return super.doAfterBody();
		
//		return EVAL_BODY_AGAIN;
		
	}

	@Override
	public int doEndTag() throws JspException {
		System.out.println("=======doEndTag=======");
//		return super.doEndTag();

		return SKIP_PAGE;
	}

}

四、案例

1、if

助手类:

package com.tgq.jsp01;

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;

/**
 * if标签 
 * 需要获取到的是满足条件的结果值,那么改标签就有一个属性,属性值是boolean
 * 
 * @author tgq
 *
 */
public class IfTag extends BodyTagSupport {
	private boolean test;

	public boolean isTest() {
		return test;
	}

	public void setTest(boolean test) {
		this.test = test;
	}

	@Override
	public int doStartTag() throws JspException {
		// 如果满足条件,就打印标签体==》 doStartTag的返回值EVAL_BODY_INCLUDE
		// 不满足条件,就不输出标签体==》 doStartTag的返回值SKIP_BODY
		return test ? EVAL_BODY_INCLUDE : SKIP_BODY;
	}

}

tld文件内容:

<?xml version="1.0" encoding="UTF-8" ?>

<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd"
    version="2.0">
    
  <description>JSTL 1.1 core library</description>
  <display-name>JSTL core</display-name>
  <tlib-version>1.1</tlib-version>
  <!-- 一个名字 -->
  <short-name>t</short-name>
  <!-- 路径 -->
  <uri>http://jsp.tgq.cn</uri>

<tag>
	<!-- 标签库名 -->
    <name>demo1</name>
    <!-- 对应的助手类 -->
    <tag-class>com.tgq.jsp01.DemoTag01</tag-class>
    <!-- 代表JSP标签 -->
    <body-content>JSP</body-content>
    <!-- <attribute>
    自定义JSP标签的属性名称 
        <name>var</name>
        该属性是否必填
        <required>false</required>
        该属性值是否支持表达式
        <rtexprvalue>false</rtexprvalue>
    </attribute> -->
</tag>


  <tag>
    <name>if</name>
    <tag-class>com.tgq.jsp01.IfTag</tag-class>
    <body-content>JSP</body-content>
    <attribute>
        <name>test</name>
        <required>true</required>
        <rtexprvalue>true</rtexprvalue>
    </attribute>
  </tag>
  </taglib>

JSP页面:

<%@ page language="java" contentType="text/html; charset=UTF-8"
	pageEncoding="UTF-8"%>
<%@taglib uri="http://jsp.tgq.cn" prefix="t"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>IfDemo</title>
</head>
<body>
	<t:if test="true">true</t:if>
	<t:if test="false">false</t:if>
</body>
</html>

输出结果:

2、out

首先如果我们要输出结果的话是不是要设置一个Set的标签去存值然后进行一个输出。

剩下的基本都是源代码了,你们可以cv参考

tld

<?xml version="1.0" encoding="UTF-8" ?>

<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd"
    version="2.0">
    
  <description>JSTL 1.1 core library</description>
  <display-name>JSTL core</display-name>
  <tlib-version>1.1</tlib-version>
  <!-- 一个名字 -->
  <short-name>t</short-name>
  <!-- 路径 -->
  <uri>http://jsp.tgq.cn</uri>

<tag>
	<!-- 标签库名 -->
    <name>demo1</name>
    <!-- 对应的助手类 -->
    <tag-class>com.tgq.jsp01.DemoTag01</tag-class>
    <!-- 代表JSP标签 -->
    <body-content>JSP</body-content>
    <!-- <attribute>
    自定义JSP标签的属性名称 
        <name>var</name>
        该属性是否必填
        <required>false</required>
        该属性值是否支持表达式
        <rtexprvalue>false</rtexprvalue>
    </attribute> -->
</tag>


  <tag>
    <name>if</name>
    <tag-class>com.tgq.jsp01.IfTag</tag-class>
    <body-content>JSP</body-content>
    <attribute>
        <name>test</name>
        <required>true</required>
        <rtexprvalue>true</rtexprvalue>
    </attribute>
  </tag>
  
  
  <tag>
    <name>set</name>
    <tag-class>com.tgq.jsp01.SetTag</tag-class>
    <body-content>JSP</body-content>
    <attribute>
        <name>var</name>
        <required>true</required>
        <rtexprvalue>false</rtexprvalue>
    </attribute>
    <attribute>
        <name>value</name>
        <required>true</required>
        <rtexprvalue>true</rtexprvalue>
    </attribute>
  </tag>
  
  
  <tag>
    <name>out</name>
    <tag-class>com.tgq.jsp01.OutTag</tag-class>
    <body-content>JSP</body-content>
    <attribute>
        <name>value</name>
        <required>true</required>
        <rtexprvalue>true</rtexprvalue>
    </attribute>
  </tag>
</taglib>

JSP页面:

<%@ page language="java" contentType="text/html; charset=UTF-8"
	pageEncoding="UTF-8"%>
<%@taglib uri="http://jsp.tgq.cn" prefix="t"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>IfDemo</title>
</head>
<body>
	<t:set var="name" value="lsqwertyuiop"></t:set>
	<t:out value="${name }"></t:out>
</body>
</html>

set助手类:

package com.tgq.jsp01;

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;

/**
 * 数据标签:存储数据 
 * 作用:pagecontext、request、session、application(servletContext)
 * 
 * 
 * @author tgq
 *
 */
public class SetTag extends BodyTagSupport {
	private String var;
	private Object value;

	public String getVar() {
		return var;
	}

	public void setVar(String var) {
		this.var = var;
	}

	public Object getValue() {
		return value;
	}

	public void setValue(Object value) {
		this.value = value;
	}

	@Override
	public int doStartTag() throws JspException {
		// 要存储数据,以键值对的形式存储,分析得出该标签有两个属性
		pageContext.setAttribute(var, value);
		return super.doStartTag();
	}
}

out助手类:

package com.tgq.jsp01;

import java.io.IOException;

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.JspWriter;
import javax.servlet.jsp.tagext.BodyTagSupport;

/**
 * 将数据输出到前台,首先拿到输出流
 * 
 * @author tgq
 *
 */
public class OutTag extends BodyTagSupport {
	private Object value;

	public Object getValue() {
		return value;
	}

	public void setValue(Object value) {
		this.value = value;
	}

	@Override
	public int doStartTag() throws JspException {
		JspWriter out = pageContext.getOut();
		try {
			out.print(value);
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}

		return super.doStartTag();
	}
}

希望对大家有用,感谢!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/693022.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

# rust abc(6): 字符串的简单使用

文章目录 1. 目的2. 数据类型2.1 str 类型2.2 标准库 String 类型 3. 常用 API3.1 len() 方法3.2 is_empty() 方法3.3 starts_with() 方法3.4 find() 方法 4. References 1. 目的 学习 Rust 语言中的字符串&#xff0c; 包括数据类型&#xff0c; 常用 API。 2. 数据类型 Ru…

新手入门:从零搭建vue3+webpack实战项目模板

搭建一个 vue3 webpack5 element-plus 基本模板 &#xff08;vue3 webpack5 从零配置项目&#xff09;。 本项目结构可以作为实战项目的基本结构搭建学习&#xff0c;作为刚学习完vue还没有实战项目经验的小伙伴练习比较合适。 项目地址&#xff1a; GitHub&#xff1a;ht…

如何将手写笔记转换成电子版格式?

记笔记是一种非常有效的学习方法。它不仅可以帮助我们加深对所学内容的理解&#xff0c;还能让我们收集更多有用的信息&#xff0c;以方便后续的查看和复习。不过&#xff0c;用传统的纸质笔记本记录笔记存在一定的弊端&#xff0c;比如说不易保存、不易携带等等。所以&#xf…

Mac下的java.io.FileNotFoundException: ~/Desktop/a.sql (No such file or directory)

【问题】&#xff1a; 今天在运行一个文件读取的Demo时&#xff0c;报如下错误: java.io.FileNotFoundException: ~/Desktop/a.sql (No such file or directory)如下图所示 &#xff1a; 可是这个文件命名可以通过终端窗口访问到啊&#xff1f; 【解决方案】&#xff…

STM32外设系列—HC-SR04(超声波)

文章目录 一、超声波测距基本原理二、超声波传感器简介三、HC-SR04测距实现思路四、超声波测距程序实现4.1 HC-SR04初始化程序4.3 TIM开关程序4.4 获取定时时间4.5 计算测量距离4.6 宏定义 五、应用实例六、拓展应用 一、超声波测距基本原理 超声波测距的原理非常简单&#xf…

高压放大器在压电陶瓷驱动器中的应用

高压放大器是一种将低电压信号放大成高电压信号的电子设备。它广泛运用于各种领域&#xff0c;如医疗、工业、军事以及科学研究。压电陶瓷驱动器是一种利用压电效应来驱动机械运动的装置。这两种设备经常被用于控制和操作许多不同类型的系统。 压电陶瓷是一种能够将电能转化为机…

监控摄像头的像素200万,400万,800万都是什么意思,200万像素、400万像素、800万像素是如何换算出来的?

一、像素 像素&#xff08;Pixel&#xff09;是用来表示图像分辨率的单位&#xff0c;数字越大&#xff0c;表示图像中的细节可以更精细地展现。当我们谈论监控摄像头的像素时&#xff0c;通常指的是摄像头图像传感器上的像素数量。像素的数量可以通过传感器上的横向像素数乘以…

win如何使用OpenSSL生成自签名证书,使 http 升级为 https

win如何使用OpenSSL生成自签名证书&#xff0c;使 http 升级为 https 前言 HTTPS其实就是HTTP over SSL&#xff0c;也就是让HTTP连接建立在SSL安全连接之上。 创建自签名证书需要安装openssl。参考本文安装OpenSSL部分。 使用OpenSSL生成自签名证书的步骤&#xff1a;参考…

python spider 爬虫 之 Selenium 系列 (-) Selenium

京东的 seckill 秒杀 专区 用 urllib 是获取不到的 回顾一下urllib 爬虫 # urllib 爬虫 from urllib import request headers {} url # 请求定制 req request(urlurl, headers headers) # 模拟请求 response request(req) content response.read().decode(utf-…

windows下安装Visual Studio + CMake+OpenCV + OpenCV contrib

目录 1 安装visual studio 2 安装CMake 3 OpenCV源码安装 3.1 OpenCV源码下载 3.2 OpenCV contrib源码下载 3.3 安装OpenCV 3.4 安装OpenCV-crontrib 3.5 VS生成代码 4 环境配置 最近在研究windows系统上部署安装目标检测算法&#xff0c;需要用到OpenCV软件&#xff…

智能指针+拷贝构造+vector容器+多态引起的bug

今天在调试一段代码的时候&#xff0c;VC编译提示&#xff1a; error C2280: “T485CommCtrlPara::T485CommCtrlPara(const T485CommCtrlPara &)”: 尝试引用已删除的函数 函数执行部分如下&#xff1a; 看意思是这个pComm485Pro已经消亡了&#xff0c;后续push_back到ve…

高速电路设计系列分享-信号链精度分析(中)

目录 概要 整体架构流程 技术名词解释 技术细节 1.直流无源误差 小结 概要 提示&#xff1a;这里可以添加技术概要 在任何设计中&#xff0c;信号链精度分析都可能是一项非常重要的任务&#xff0c;必须充分了解。之前&#xff0c; 我们讨论了在整个信号链累积起来并且最终会…

统一日志处理----AOP/面向切面编程

AOP Aspect Oriented Programing&#xff1a;面向切面编程 AOP是对OOP的补充&#xff0c;进一步提高编程的效率 AOP的常见使用场景有&#xff1a;权限检查、记录日志、事务管理等 如下图所示结构&#xff0c;每个模块都含有相同的系统需求&#xff0c;而这些需求和模块本身的功…

Flutter进阶-动画详解

目录 动画类别 一、隐式&#xff08;全自动&#xff09;动画 二、显式动画&#xff08;手动控制&#xff09; 三、其他动画(CustomPainter) 动画类别 Flutter 中有多种类型的动画&#xff1a; 隐式动画&#xff1a;通过更改部件属性自动触发的预定义动画&#xff0c;例如 …

什么是cookie

1、cookie是什么 Cookie&#xff0c;有时也用其复数形式Cookies。类型为“小型文本文件”&#xff0c;是某些网站为了辨别用户身份&#xff0c;进行Session跟踪而储存在用户本地终端上的数据&#xff08;通常经过加密&#xff09;&#xff0c;由用户客户端计算机暂时或永久保存…

Python强类型编程

Python是一门强类型的动态类型语言&#xff0c;具体如下特性&#xff1a; 可以动态构造脚本执行、修改函数、对象类型结构、变量类型但不允许类型不匹配的操作 第一个例子体现动态性&#xff1a;用字符串直接执行代码&#xff0c;动态构建了一个函数并执行&#xff0c;甚至给…

力扣744.寻找比目标字母大的最小字符(java暴力查找法,二分查找法)

题目描述&#xff1a; 给你一个字符数组 letters&#xff0c;该数组按非递减顺序排序&#xff0c;以及一个字符 target。letters 里至少有两个不同的字符。 返回 letters 中大于 target 的最小的字符。如果不存在这样的字符&#xff0c;则返回 letters 的第一个字符。 [外链…

岭回归(Ridge)不同alpha值对归回结果的影响

对于有些矩阵&#xff0c;矩阵中某个元素的一个很小的变动&#xff0c;会引起最后计算结果误差很大&#xff0c;这种矩阵称为“病态矩阵”。有些时候不正确的计算方法也会使一个正常的矩阵在运算中表现出病态。对于高斯消去法来说&#xff0c;如果主元&#xff08;即对角线上的…

亚马逊测评:如何有效使用IP和养号设备环境

随着网络科技的崛起&#xff0c;越来越多的本土企业入驻亚马逊电子商务平台上&#xff0c;这导致了对产品评价需求的激增。然而&#xff0c;评价并非随意进行&#xff0c;它需要多方面的资源&#xff0c;并需要密切注意一些重要环节。以下是我分享给大家一些宝贵的知识&#xf…

如何实现敏捷交付中的自动化测试优化

在提及自动化测试的时候&#xff0c;很多人会把工具的使用等同于自动化测试。自动化测试应该是一个策略性的系统化工程&#xff0c;不只有自动化工具。自动化测试要发挥其频繁快速的质量反馈作用&#xff0c;还需要团队从文化和技术上去建设和学习。 提到敏捷交付&#xff0c;…