本篇文章介绍自定义标签,可能在工作中很少涉及到自己来定义一个标签库,因为我们基本上都是使用的大神写的标签库,基本上直接使用即可,但是从自身的发展来看,通往高级程序员的道路上,开发框架就需要大量的使用到标签库技术。本文将从以下几个方面介绍自定义标签库的基本知识点:
- 背景以及作用
- 开发简单的标签
- 开发带属性的标签
- 开发带标签体的标签
- 以页面片段为属性的标签
- 具有动态属性的标签
一、标签库有什么作用
自定义标签库是一种优秀的表现层技术,之前介绍的MVC模式,我们使用jsp作为表现层,但是jsp语法嵌套在html页面,美工还是很难直接参与开发,并且jsp脚本和html代码耦合在一起,维护成本较高。我们能不能开发一套和html风格类似并且能完成jsp脚本功能的标签来解决这种低效的协作方式呢?于是标签库就诞生了。
(但是,现在有了更好的解决方案:接口编程前端用vue.js,react.js,Angular等,我估计之后,标签在次功能上也会逐渐被取代,就和几十年前的table布局页面一样,现在已经没人用了)
这是Java中标签规范的继承体系,实现Tag接口的我们叫做传统式标签库开发,这种开发模式略显发复杂,基本已经被SimpleTag式的简单式开发标签库给取代了。Java中提供了一个默认的实现类SimpleTagSupport来实现自定义标签,我们只要继承此类即可。
生命周期:
二、开发一个最简单的标签库
开发一个自定义标签库的过程如下:
- 开发自定义标签处理类
- 创建*.tld文件,每个此文件对应一个标签库,标签库中可以由多个标签
- 在jsp页面使用标签
首先我们先从自定义标签处理类开始,正如上文所说,这个类只有继承了SimpleTagSupport这个类可以省去省去重写SimpleTag接口中的一些方法。我们说个doTag()这个方法很重要,这个方法类似于我么main方法一样,当jsp页面加载到我们定义的标签的时候就会过来调用这个方法。
public class MyTag extends SimpleTagSupport { @Override public void doTag() throws JspException, IOException { getJspContext().getOut().write("hello walker"); } }
这是一个简单的标签处理类,具体的细节暂时不用关心,只需要知道,它负责向jsp页面输出字符串即可。下面我们看看第二步,创建*tld文件。这个文件我们没有必要重新写一遍,到Tomcat服务器上的webapps/examples/WEB-INF/jsp2中复制一个过来,修改名字存放到我们的项目中WEB-INF的任意子路径下。删除一些标签成如下内容:
我们看到这是一个XML文件,根元素为taglib,而taglib主要有以下几个子元素:
- description //描述信息
- tlib-version //指定标签库的版本号,基本不用我们操心
- short-name //指定标签库的短名字,也是没什么用
- uri //这是一个重要的子元素,是该标签库的唯一标识
- tag //看名字就知道,这是定义标签的子元素,很重要
对于taglib这个根元素,我们主要关心他下面的uri和tag两个子元素,一个标签库可以由多个标签,也就是可以有多个tag标签。关于tag标签,主要有以下几个子元素:
- description //描述信息
- name //该标签的唯一标识,很重要
- tag-class //指定了处理该标签的类,也就是使用该标签谁给我返回结果
- body-content //标签体,后面详说,很重要
- attribute //属性,后面介绍,很重要
对于以上标签大家可能已经知道什么意思,但是具体用在什么地方可能不清楚,本小节的最后会综合三个步骤自定义一个简单的标签。接下来介绍在jsp页面是如何使用标签。
使用标签库也是有两个步骤,首先导入标签库,然后引用标签。我们使用taglib编译指令导入标签库,具体格式如下:
<%@ taglib uri="tld文件中指定的唯一标识" prefix="指定标签前缀"%>
我们看到这个导入标签库的编译指令主要有两个属性,一个是用于定位我们已经写好的标签库,定位的方法就是读取每个tld文件中的URI元素的值,prefix用于指定我们使用标签时的前缀,等用的时候就很容易理解了,现在解释反而不容易说清楚。我们使用标签的格式如下:
<刚刚指定的前缀 :标签名 />
标签名就是我们标签库中每个tag都会有的name的值,这指定了该语句是引用的那个标签。下面我们通过具体的例子直观感受下。
<%@ page contentType="text/html;charset=UTF-8" language="java" %> <%@taglib uri="mytid" prefix="mytags"%> <html> <head> <title>title> head> <body> <mytags:hello /> body> html>
<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>A tag library exercising SimpleTag handlers.description> <tlib-version>1.0tlib-version> <short-name>SimpleTagLibraryshort-name> <uri>mytiduri> <tag> <description>Outputs a colored tiledescription> <name>helloname> <tag-class>Test.MyTagtag-class> <body-content>emptybody-content> tag> taglib>
public class MyTag extends SimpleTagSupport { @Override public void doTag() throws JspException, IOException { getJspContext().getOut().write("hello walker"); } }
结果如下:
这就完成了一个最简单的标签库的定义和使用的过程,首先我们在index.jsp页面通过URI引入mytag标签库,指定了使用该标签库的前缀为mytags,然后
三、开发带属性的标签
假如我们通过拦截器获取了从数据库查出来的一个结果集,我们此处希望调用标签来将结果集以表格的形式输出来,此时我们的这个结果集又该如何传到标签处理类中呢?这时我们可以使用属性。具体看代码:
<%@ page contentType="text/html;charset=UTF-8" language="java" %> <%@taglib uri="mytid" prefix="mytag"%> <html> <head> <title>title> head> <body> <% HashMap<String,Integer> maps = new HashMap<String, Integer>(); maps.put("李四",53); maps.put("张三",23); maps.put("walker",22); pageContext.setAttribute("map",maps); %> <table> <mytag:hello map="map"/> table> body> html>
<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>A tag library exercising SimpleTag handlers.description> <tlib-version>1.0tlib-version> <short-name>SimpleTagLibraryshort-name> <uri>mytiduri> <tag> <description>Outputs a colored tiledescription> <name>helloname> <tag-class>Test.MyTagtag-class> <body-content>emptybody-content> <attribute> <name>mapname> <required>truerequired> <fragment>truefragment> attribute> tag> taglib>
public class MyTag extends SimpleTagSupport { private String map; public String getMap(){ return this.map; } public void setMap(String map){ this.map = map; } @Override public void doTag() throws JspException, IOException { HashMapmaps = (HashMap )(getJspContext().getAttribute(map)); Object[] array = maps.keySet().toArray(); for (String str : maps.keySet()){ getJspContext().getOut().write(" "); getJspContext().getOut().write(" "); } } }"); getJspContext().getOut().write(str); getJspContext().getOut().write(" "); getJspContext().getOut().write(""); getJspContext().getOut().write(""+maps.get(str)); getJspContext().getOut().write(" "); getJspContext().getOut().write("
我们首先先从index.jsp页面看起,首先我们定义了一个HashMap用来存放一个简单的个人信息,键为姓名值为年龄。最后我们设置共享范围为当前page。然后
这个类定义了私有属性map,和我们的tld文件中的属性名是一致的。getJspContext().getAttribute(map),首先是获得了调用该标签的jsp页面的pageContext,这就是方法getJspContext的返回值,因为我们在jsp页面设置了一个共享数据(maps),于是我们同名名字获取该对象,这里的map就是我们的私有属性,他的值被自动赋值了,具体的值就是jsp页面传入的参数。后面的代码就很简单了,循环输出数据到jsp页面上。
稍微理一下思路,这种带属性的标签使用其实和无属性差不多,都是先引入了标签库,加载标签的的时候通过URI找到对应的标签库,只不过这次将一个字符串赋值给了tld中attribute元素中名为map的属性,然后跳向对应的标签处理类,顺便把map属性的值自动赋值处理类中的私有属性,然后执行输出代码。其中需要注意的是属性名一定要统一,另外,如果标签的属性值是8种基本数据类型,那么在JSP页面在传递字符串时,JSP引擎会自动转换成相应的类型,但如果标签的属性值是复合数据类型,那么JSP引擎是无法自动转换的。对于传递非基本数据类型的操作,后续文章会介绍.
转:
https://www.cnblogs.com/yangming1996/p/6679494.html
https://blog.csdn.net/yy339452689/article/details/78520692