JSP1.x 自定义标签

Tag接口

任何一个标签都对应着一个java类,该类必须实现Tag接口,JSP遇到一个标签后后,将通过一个tld文件查找该标签的实现类,并运行该类的相关方法

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.PageContext;
import javax.servlet.jsp.tagext.Tag;
 
public class TagTestimplements Tag {
      private Tag parent;
      private PageContext pageContext;
 
      @Override
      public int doEndTag() throws JspException{  //标签结束时执行
             JspWriter out =pageContext.getOut();
             try{
                    out.println("……");
             }catch(IOExceptione){
                    throw new JspException(e);
             }
             return EVAL_PAGE;
      }
 
      @Override
      public int doStartTag() throwsJspException {  //标签开始时执行
             return SKIP_BODY;
      }
 
      @Override
      public Tag getParent() {
             // TODO Auto-generated method stub
             return null;
      }
 
      @Override
      public void release() {
             // TODO Auto-generated method stub
 
      }
 
      @Override
      public void setPageContext(PageContextarg0) {
             // TODO Auto-generated method stub
 
      }
 
      @Override
      public void setParent(Tag arg0) {
             // TODO Auto-generated method stub
 
      }
 
}


doStartTag方法可以返回两种参数,如果为SKIP_BODY,表示标签体内的内容不被输出,如果为EVAL_BODY_INCLUDE,则执行标签体内的代码

doEndTag也可以返回两种参数,如果为SKIP_PAGE,表示不执行标签后面的内容。如果为EVAL_PAGE,则执行标签后面的内容

此类中有两个属性,parent和pageContext,parent为该标签的父标签(或者是上一层的标签),pageContext为运行该标签的JSP页面,这两个参数都是JSP在运行时通过setter方法注射进去的

 

Tag实现类有了,还需要tld文件(TagLibrary Descriptor)

 

<?xml version="1.0" ecoding="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                 web-jsptaglibrary_2_0.xsd
      version="2.0">
 
      <tlibversion>1.0</tlibversion>
      <jspversion> 1.1</jspversion>
      <shortname>tagname</shortname>
      <uri>http://www.clf.com/tags</uri>
      <info>A simple tag test</info>
 
<tag>
      <name>tagname</name>
      <tagclass>com.chen.tags.TagTest</tagclass>
      <bodycontent>JSP</ bodycontent>
      <info>tag information</info>
</tag>
 
</taglib>

 

shortname也就是推荐使用的prefix,uri就是引用这个标签库时使用的uri

bodycontent为标签体的限制,有三种取值

empty:不允许有标签体的存在,如果有会抛出异常

JSP:允许有标签体存在,可以为JSP代码

tagdependent:允许有标签体存在,但是标签里的JSP代码不会被执行

 

如果tld文件位于/WEB-INF/下面,Tomcat会自动加载;如果位于其他位置,可以在web.xm中配置

<jsp-config>
      <taglib>
             <taglib-uri> http://www.clf.com/tags</taglib-uri>
             <taglib-location>/WEB-INF/taglib.tld</taglib-location>
      </taglib>
</jsp-config>

TagSupport

多数情况下不需要直接实现Tag接口,使用TagSupport类就可以了,此类是java提供的一个模板类,一般来说,只需要实现doStartTag方法和doEndTag方法就可以了

 

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.TagSupport;
 
 
public class Tagtest extends TagSupport {
      private int num1;
      private int num2;
 
      @Override
      public int doEndTag() throws JspException{
             try {
             this.pageContext.getOut().println("两数相加等于:"+(num1+num2));
             } catch (Exception e) {
                   
             }
             return EVAL_PAGE;
      }
 
      @Override
      public int doStartTag() throwsJspException {
             return super.doStartTag();
      }
 
      public int getNum1() {
             return num1;
      }
 
      public void setNum1(int num1) {
             this.num1 = num1;
      }
 
      public int getNum2() {
             return num2;
      }
 
      public void setNum2(int num2) {
             this.num2 = num2;
      }
 
}

 

配置文件

 

<tag>
      <name>add</name>
      <tagclass>com.chen.tags.TagTest</tagclass>
      <bodycontent>JSP</bodycontent>
      <info>tag information</info>
      <attribute>
             <name>num1</name>
             <required>true</required >
             <rtexprvalue>true</rtexprvalue >
      </attribute>
      <attribute>
             <name>num2</name>
             <required>true</required >
             <rtexprvalue>true</rtexprvalue >
      </attribute>
</tag>

 

name代表属性的名字,required代表是否是必需的,rtexprvalue代表此属性是否允许使用EL表达式

以上这个标签这样使用:

<taglib:addnum1=”2” num2=”3”/>

则页面会输出2和3 的相加之和

 

BodyTagSupport

BodyTagSupport是Tag接口的子类,专门处理带标签体的标签

public class ToLowerCaseTag extends BodyTagSupport{
      public int doEndTag throws JspException{
             //获取标签体内的代码
             String contend = this.getBodyContent().getString();
      try{
             this.pageContent.getOut().println(content.toLowerCase());
      }catch(Exception e){   }
             return EVAL_PAGE;
      }
}

只要在setBodyContent()方法之后被调用的方法中,都可以使用getBodyContent()方法获取标签体的内容,一般为doAfterBody()或者doEndTag()

BodyTagSupport的执行流程如下:

1.当容器创建一个新的标签实例后,通过setPageContext来设置标签的页面上下文.

2.使用setParent方法设置这个标签的上一级标签,如果没有上一级嵌套,设置为null.

3.设置标签的属性,这个属性在标签库描述文件中定义,如果没有定义属性,就不调用此类方法.

4.调用doStartTag方法,这个方法可以返回EVAL_BODY_INCLUDE和SKIP_BODY,当返回EVAL_BODY_INCLUDE时,就计算标签的body,如果返回SKIP_BODY,就不再计算标签的body,如果为EVAL_BODY_BUFFERED,则不会输出,而是将标签体内容通过setBodyContent()方法注射到标签类中,然后就可以使用getBodyContent()取得标签体的内容

5.调用setBodyContent设置当前的BodyContent.

6.调用doInitBody,如果计算BodyContent时需要进行一些初始化,就在这个方法中进行.

7.每次计算完Body后调用doAfterBody,如果返回EVAL_BODY_AGAIN,表示继续计算一次Body,直到返回SKIP_BODY才继续往下执行.

8.调用doEndTag方法,这个方法可以返回EVAL_PAGE或者SKIP_PAGE,当返回EVAL_PAGE时,容器将在标签结束时继续计算JSP页面其他的部分;如果返回SKIP_PAGE,容器将在标签结束时停止计算JSP页面其他的部分.

9.调用release()方法释放标签程序占用的任何资源。

 技术分享

 

public class LoopTag extends BodyTagSupport{
      private int times;
     
      public int doStartTag() throwsJspException{
             times = 5;
             return super.doStartTag();
      }
 
      public int doAfterBody() throwsJspException{
             if(times-- >0){ 只要times大于零继续循环,同时times自减
                    try{
                    this.getPreviousOut().println(this.getBodyContent().getString());
                    }catch(Exception e){  }
                    return EVAL_BODY_AGAIN;  //继续执行
             }else{
                    return SKIP_BODY;
             }
      }
}

以上定义了一个循环标签

<taglib:loop>loop.</taglib:loop>

会得到如下输出:

loop.

loop. loop.

loop. loop. loop.

loop. loop. loop.loop.

loop. loop. loop.loop. loop.

 

实际上,doAfterBody方法内的输出是写到bodyContent缓存中的

BodyTagSupport中,里面的标签可以通过getparent()方法获得上层的标签

 

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。