JSP20Examples-自定义标签

在JSP2.0 中,对于自定义的标签有两种实现方法,实现接口或者继承现有的类
如下图,标注蓝色的是接口,其它是标签类(SimpleTagSupport只在JSP2.0中才有)

 

JspTag (蓝色为接口)

 

Tag ——〉IterationTag

SimpleTag

 

BodyTag

 

TagSupport

BodyTagSupport extends TagSupport implements BodyTag

SimpleTagSupport


在以上接口和类中,定义了一些静态常量,如下:
Tag中定义:
SKIP_BODY = 0;          // 不处理标签体,直接调用doEndTag()方法
EVAL_BODY_INCLUDE = 1;   // 解析标签体,但绕过doInitBody()和setBodyContent()方法
SKIP_PAGE = 5;          // 不解析标签后面的JSP内容
EVAL_PAGE = 6;          // 解析标签后,继续解析标签后面的JSP内容
IterationTag中定义:
EVAL_BODY_AGAIN = 2;
BodyTag中定义:
EVAL_BODY_TAG = 2;         // deprecated
EVAL_BODY_BUFFERED = 2;   
特别的,对于EVAL_BODY_AGAIN和EVAL_BODY_BUFFERED:
在doAferBody中返回SKIP_BODY,表示终止标记正文处理;若返回的是EVAL_BODY_BUFFERED,将会再一次调用doAferBody方法,重新处理标记正文,直到返回SKIP_BODY为止。

下面是自定义tag的执行过程(由上至下),对于以上各常量的实际运用为:
注意其中的doInitBody/setBodyContent方法在自定义标签实现了BodyTag接口或继承BodyTagSupport才可以使用

Tag方法

可返回的静态常量

doStartTag

SKIP_BODYEVAL_BODY_INCLUDE

EVAL_BODY_AGAIN/EVAL_BODY_BUFFERED

doInitBody

做标签一些初始化工作,无返回值

setBodyContent

doInitBody之后执行,使用setBodyContent得到JSP页面中标签体之间内容

doAfterBody

最终必须返回SKIP_BODY,否则可能导致OutOfMemoryError,可参考上面

doEndTag

SKIP_PAGE/EVAL_PAGE


附①示例代码如下:
public int doAfterBody() throws JspException {
try { 
   this.pageContext.getOut().write("<br>");
} catch (IOException e) { 
      e.printStackTrace();
}
if(cou>1){ 
   cou--; 
   return this.EVAL_BODY_AGAIN;
}else{ 
   return this.SKIP_BODY;   // 最终必须返回SKIP_BODY
}
}

自定义标签的开发包括:
1. 开发标签的处理程序(java类)
2. .tld文件中指定标签使用的类
3. 在web.xml中指定JSP中使.tld(标签库描述文件)文件的位置。
在.tld文件中
<tag> 
    <name>out</name> 
    <tag-class>org.apache.taglibs.standard.tag.el.core.OutTag</tag-class>
<body-content>JSP</body-content>
<attribute> 
        <name>value</name> 
        <required>true</required> 
        <rtexprvalue>false</rtexprvalue>
</attribute>
</tag>
body-content:
根据web-jsptaglibrary_2_0.xsd(位于servlet-api.jar包($TOMCAT_HOME\common\lib)中的\javax\servlet\resources下,其中web.xml验证时所需要的xsd文件都位于此resources目录下),body-content的值有下面4种: 
    <xsd:enumeration value="tagdependent"/> 
    <xsd:enumeration value="JSP"/> 
    <xsd:enumeration value="empty"/> 
    <xsd:enumeration value="scriptless"/>
tagdependent:标签体内容直接被写入BodyContent,由自定义标签类来进行处理,而不被JSP容器解释,
如下:
<test:myList>
select name,age from users
</test:myList>
JSP:接受所有JSP语法,如定制的或内部的tag、scripts、静态HTML、脚本元素、JSP指令和动作。如:
<my:test> 
    <%=request.getProtocol()%>     // ②
</my:test>
具体可参考后面附源码。
empty:空标记,即起始标记和结束标记之间没有内容。
下面几种写法都是有效的,
<test:mytag />
<test:mytag uname="Tom" />
<test:mytag></test:mytag>
scriptless:接受文本、EL和JSP动作。如上述②使用<body-content> scriptless </body-content>则报错,具体可参考后面附源码。
rtexprvalue:
由请求时表达式来指定属性的值,默认为false,如下必须设置为true:
<test:welcome uname="<%=request.getParameter("username") %>" />
附body-content为JSP/scriptless时标签体可以接受的代码(jasper-compiler.jar包($TOMCAT_HOME\common\lib)中的\org\apache\jasper\compiler\Parser.java中):
JSP局限性比较小,在body-content可以使用Scriptless的地方都可以用JSP代替,反之则不可

你可能感兴趣的:(apache,tomcat,Web,jsp,servlet)