自定义JSP标签就是程序员定义的一种JSP标签,这种标签把那些信息显示逻辑封装在一个单独的Java类中,通过一个XML文件来描述它的使用。当页面中需要使用类似的显示逻辑时,就可以在页面中插入这个标签,从而完成相应的功能。
使用自定义标签,可以分离程序逻辑和表示逻辑,将Java代码从HTML中剥离,便于美工维护页面;自定义标签也提供了可重用的功能组件,能够提高工程的开发效率。
自定义标签主要用于移除Jsp页面中的java代码。
当一个含有自定义标签的JSP页面被JSP引擎(Web容器)转译成Servlet时,JSP引擎遇到自定义的标签,会把这个自定义标签转化成对一个称为“标签处理类”的调用。之后,当这个JSP页面被执行时,JSP引擎就会调用这个“标签处理类”对象,并执行其内部定义的相应操作方法,从而完成相应的功能。
使用Java处理类来开发自定义JSP标签时,主要分为下几个步骤。
创建标签的处理类(Tag Handle Class)。这个类实现Tag接口,用来定义标签的行为,并在JSP引擎遇到自定义标签时调用执行。
创建标签库描述 (tld) 文件(Tag Library Descriptor File),在tld文件中对标签处理器类进行描述。
在JSP文件中用taglib指令引入标签库,然后使用标签库描述文件中指定的标签名来使用它。
JSP引擎将遇到自定义标签时,首先创建标签处理器类的实例对象,然后按照JSP规范定义的通信规则依次调用它的方法。
1、public void setPageContext(PageContext pc), JSP引擎实例化标签处理器后,将调用setPageContext方法将JSP页面的pageContext对象传递给标签处理器,标签处理器以后可以通过这个pageContext对象与JSP页面进行通信。
2、public void setParent(Tag t),setPageContext方法执行完后,WEB容器接着调用的setParent方法将当前标签的父标签传递给当前标签处理器,如果当前标签没有父标签,则传递给setParent方法的参数值为null。
3、public int doStartTag(),调用了setPageContext方法和setParent方法之后,WEB容器执行到自定义标签的开始标记时,就会调用标签处理器的doStartTag方法。
4、public int doEndTag(),WEB容器执行完自定义标签的标签体后,就会接着去执行自定义标签的结束标记,此时,WEB容器会去调用标签处理器的doEndTag方法。
5、public void release(),通常WEB容器执行完自定义标签后,标签处理器会驻留在内存中,为其它请求服务器,直至停止web应用时,web容器才会调用release方法。
开发人员在编写Jsp页面时,经常还需要在页面中引入一些逻辑,例如:
控制jsp页面某一部分内容是否执行。
控制整个jsp页面是否执行。
控制jsp页面内容重复执行。
修改jsp页面内容输出。
自定义标签除了可以移除jsp页面java代码外,它也可以实现以上功能。
tld文件中的四种标签体类型
EMPTY JSP scriptless tagdepentend
自定义标签的实现
1.编写一个实现tag接口的类
public class ViewIPTag extends TagSupport { @Override public int doStartTag() throws JspException { HttpServletRequest request = (HttpServletRequest) this.pageContext.getRequest(); JspWriter out =this.pageContext.getOut(); String ip = request.getRemoteAddr(); try { out.println(ip); } catch (IOException e) { throw new RuntimeException(e); } return super.doStartTag(); } }
2.建立tld文件,对标签处理类进行描述。(放在WEB-INF文件夹下)
(C:\apache-tomcat-6.0.18\webapps\examples\WEB-INF\jsp2)
<?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>A tag library exercising SimpleTag handlers.</description> <tlib-version>1.0</tlib-version> <short-name>hbsi</short-name> <uri>http://www.hbsi.com</uri> <tag> <name>viewIP</name> <tag-class>com.hbsi.web.tag.ViewIPTag</tag-class> <body-content>empty</body-content> </tag> </taglib>
3.在jsp文件中使用自定义的标签
<%@ taglib uri="http://www.hbsi.com" prefix="hbsi" %> <body> 您的IP地址是:<hbsi:viewIP/> </body>
JSP 1.1和1.2规范中常用的接口主要有以下3个。
标签库描述符文件是一个以“.tld”结尾的标准XML文档,用来记录一个标签库内拥有哪些标签、每个标签包含哪些属性。
以下是一个JSP 2.0规范的标签库描述文件的内容:
<taglib>元素是标签库描述符的根元素,它包含12个子元素,详细介绍如下。
(1)<description>:标签库的一个文本描述。
(2)<tlib-version>:指定标签库的版本。
(3)<short-name>:为标签定义简短的名字,在taglib指令中可作为首选的前缀名使用。
(4)<uri>:定义一个URI,用于唯一地标识此标签库。
(5)<tag>:用于指定自定义标签的相关信息。
(6)<display-name>:为标签库指定一个简短的别名。
(7)<small-icon>:为标签库指定大小为16×16的小图标(gif或jpeg格式),该图标可在图形界面工具中显示。
(8)<large-icon>:为标签库指定大小为32×32的大图标(gif或jpeg格式),该图标可在图形界面工具中显示。
(9)<validator>:为标签库提供一个验证器。
(10)<listener>:为标签库提供一个监听器。
(11)<tag-file>:用于描述标签文件。
(12)<function>:用于指定在表达式语言中使用的函数。
Tag 接口与 BodyTag 接口
Tag 接口是十分重要的接口 它定义了 Tag Handler 与 JSP page implementation class 之 间通讯的基础协议 主要用于处理 Tag 自身的 Action。
TagSupport 类与 BodyTagSupport 类
TagSupport 类是十分重要的类 它实现了 Tag 接口中定义的方法 在 TagSupport 类中 实现的方法如下所示
TagInfo 类 TagExtraInfo 类
TagInfo 类和 TagExtraInfo 类映射了 Tag 的描述信息 而 Tag 的描述信息在 Tag Library 的 TLD 文件中定义 所以 TagInfo 类和 TagExtraInfo 类正是通过读取 TLD 文件 从而获取 Tag 的描述信息
TagLibraryInfo 类与 TagAttributeInfo 类
TagLibraryInfo 类映射了 Tag Library 的详细信息 TagLibraryInfo 类其实也是通过读取 TLD 文件 才能够获取 Tag Library 的信息
BodyContent 类
BodyContent 类代表了 Tag Body 的内容 BodyContent 类继承自 JspWriter 类 利用它 既可以读取 Tag Body 的内容 也可以输出 Tag Body 的内容
在开发传统标签之前,需要了解清楚两个支持类(TagSupport和BodyTagSupport)的生命周期。
TagSupport类的生命周期可以用图来表示。
BodyTagSupport类的生命周期可以用图来表示。
用BodyTagSupport类开发自定义标签
BodyTagSupport类中增加了两个方法:
BodyContent类专门用于缓存标签主体返回的内容,包括静态文本以及由嵌套标签或脚本元素所创建的动态内容。