一、基本概念

1、标签(Tag)

标签是一种XML元素,通过标签可以使JSP网页变得简洁并且易于维护,还可以方便地实现同一个JSP文件支持多种语言版本。由于标签是XML元素,所以它的名称和属性都是大小写敏感的。

2、标签库(Tag library)

由一系列功能相似、逻辑上互相联系的标签构成的集合称为标签库。

3、标签库描述文件(Tag Library Descriptor)

标签库描述文件是一个XML文件,这个文件提供了标签库中类和JSP中对标签引用的映射关系。它是一个配置文件,和web.xml是类似的。

4、标签处理类(Tag Handle Class)

标签处理类是一个Java类,这个类继承了TagSupport或者扩展了SimpleTag接口,通过这个类可以实现自定义JSP标签的具体功能。

二、自定义JSP标签的格式

1、为了使到JSP容器能够使用标签库中的自定义行为,必须满足以下两个条件:

<% @ taglib prefix=”someprefix” uri=”/sometaglib” %>

1)从一个指定的标签库中识别出代表这种自定义行为的标签;
2)找到实现这些自定义行为的具体类。

第一个必需条件-找出一个自定义行为属于那个标签库-是由标签指令的前缀(Taglib Directive's Prefix)属性完成,所以在同一个页面中使用相同前缀的元素都属于这个标签库。每个标签库都定义了一个默认的前缀,用在标签库的文档中或者页面中插入自定义标签。所以,你可以使用除了诸如jsp,jspx,java,servlet,sun,sunw(它们都是在JSP白皮书中指定的保留字)之类的前缀。  

uri属性满足了以上的第二个要求。为每个自定义行为找到对应的类。这个uri包含了一个字符串,容器用它来定位TLD文件。在TLD文件中可以找到标签库中所有标签处理类的名称。

2、当web应用程序启动时,容器从WEB-INF文件夹的目录结构的META-INF搜索所有以.tld结尾的文件。也就是说它们会定位所有的TLD文件。对于每个TLD文件,容器会先获取标签库的URI,然后为每个TLD文件和对应的URI创建映射关系。

JSP页面中,我们仅需通过使用带有URI属性值的标签库指令来和具体的标签库匹配。

三、自定义JSP标签的处理过程

1、在JSP中引入标签库

<% @ taglib prefix=”taglibprefix” uri=”tagliburi” %>

2、在JSP中使用标签库标签

3Web容器根据第二个步骤中的prefix,获得第一个步骤中声明的tagliburi属性值

4Web容器根据uri属性在web.xml找到对应的元素

5.从元素中获得对应的元素的值 

6Web容器根据元素的值从WEB-INF/目录下找到对应的.tld文件

7.从.tld文件中找到与tagname对应的元素 

8.凑元素中获得对应的元素的值 

9Web容器根据元素的值创建相应的tag handle class的实例 

10. Web容器调用这个实例的doStartTag/doEndTag方法完成相应的处理。

四、创建和使用一个Tag Library的基本步骤

1、创建标签的处理类(Tag Handler Class)

2、创建标签库描述文件(Tag Library Descrptor File)

3、在web.xml文件中配置元素 4.JSP文件中引人标签库

五、TagSupport类简介

1、处理标签的类必须扩展javax.servlet.jsp.TagSupport

2TagSupport类的主要属性:

A.parent属性:代表嵌套了当前标签的上层标签的处理类;

B.pageContex属性:代表Web应用中的javax.servlet.jsp.PageContext对象;

3JSP容器在调用doStartTag或者doEndTag方法前,会先调用setPageContextsetParent方法,设置pageContextparent。因此在标签处理类中可以直接访问pageContext变量;

4、在TagSupport的构造方法中不能访问pageContext成员变量,因为此时JSP容器还没有调用setPageContext方法对pageContext进行初始化。

六、TagSupport处理标签的方法

1TagSupport类提供了两个处理标签的方法:

public int doStartTag() throws JspException
public int doEndTag() throws JspException

2doStartTag:但JSP容器遇到自定义标签的起始标志,就会调用doStartTag()方法。

doStartTag()方法返回一个整数值,用来决定程序的后续流程。

A.Tag.SKIP_BODY:表示?>…之间的内容被忽略;

B.Tag.EVAL_BODY_INCLUDE:表示标签之间的内容被正常执行。

3doEndTag:但JSP容器遇到自定义标签的结束标志,就会调用doEndTag()方法。doEndTag()方法也返回一个整数值,用来决定程序后续流程。

A.Tag.SKIP_PAGE:表示立刻停止执行网页,网页上未处理的静态内容和JSP程序均被忽略任何已有的输出内容立刻返回到客户的浏览器上。

B.Tag_EVAL_PAGE:表示按照正常的流程继续执行JSP网页。

下面来看一个实例:

1、修改web.xml文件,增加自定义标签支持。

   <jsp-config>           <taglib>               <taglib-uri>/tld/helloworld</taglib-uri>               <taglib-location>/WEB-INF/tlds/helloworld.tld</taglib-location>           </taglib>         </jsp-config>       


2、创建标签库TLD文件 tlds\helloworld.tld 。

 <?xml version="1.0" encoding="UTF-8"?>       <!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN"       "http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd">          <taglib>          <tlib-version>1.0</tlib-version><!-- 标签库的版本 -->         <jsp-version>1.2</jsp-version><!-- 这个标签库要求的JSP规范版本 -->         <short-name>mytag</short-name><!-- JSP页面编写工具可以用来创建助记名的可选名字 -->          <tag>             <name>helloworld</name><!-- 唯一标签名 -->          <tag-class>com.yd.mytag.HelloWorldTag</tag-class><!-- 标签HelloWorldTag类的完全限定名 -->            <body-content>empty</body-content><!-- 正文内容类型 -->          </tag>       </taglib>      


这里注意:web.xmlxxx.tld这两个XML文件的头信息的版本匹配很重要,否则会导致页面报错找不到标签 

3、创建标签处理程序类 HelloWorldTag(重写doStartTag和doEndTag方法)。

 package com.yd.mytag;            import java.io.IOException;       import javax.servlet.jsp.JspException;       import javax.servlet.jsp.JspTagException;       import javax.servlet.jsp.tagext.TagSupport;       /**       * TagSupport与BodyTagSupport的区别:       * 主要看标签处理类是否要读取标签体的内容和改变标签体返回的内容,如果不需要就用TagSupport,否则就用BodyTagSupport       * 用TagSupport实现的标签,都可以用BodyTagSupport来实现,因为BodyTagSupport继承了TagSupport      */       public class HelloWorldTag extends TagSupport {           private static final long serialVersionUID = 3174234039143531070L;           @Override           public int doStartTag() throws JspException {//这个方法不用所以直接返回值               return EVAL_BODY_INCLUDE;           }           @Override           public int doEndTag() throws JspException {//重点在这个方法上              try {                   pageContext.getOut().write("Hello World!");//标签的返回值                 } catch (IOException ex) {                  throw new JspTagException("错误");               }               return EVAL_PAGE;           }       }      


补充: 

doStartTag()方法是遇到标签开始时会呼叫的方法,其合法的返回值是EVAL_BODY_INCLUDESKIP_BODY,前者表示将显示标签间的文字,后者表示不显示标签间的文字。

doEndTag()方法是在遇到标签结束时呼叫的方法,其合法的返回值是EVAL_PAGESKIP_PAGE,前者表示处理完标签后继续执行以下的JSP网页,后者是表示不处理接下来的JSP网页。

doAfterBody(),这个方法是在显示完标签间文字之后呼叫的,其返回值有EVAL_BODY_AGAINSKIP_BODY,前者会再显示一次标签间的文字,后者则继续执行标签处理的下一步。

EVAL_BODY_INCLUDE:把Body读入存在的输出流中,doStartTag()函数可用。

EVAL_PAGE:继续处理页面,doEndTag()函数可用。

SKIP_BODY:忽略对Body的处理,doStartTag()doAfterBody()函数可用。

SKIP_PAGE:忽略对余下页面的处理,doEndTag()函数可用。

EVAL_BODY_BUFFERED:申请缓冲区,由setBodyContent()函数得到的BodyContent对象来处理tagbody,如果类实现了BodyTag,那么doStartTag()可用,否则非法。