JSP自定义标签(标签处理器 tld文件)
标签的形式如下,标签处理器就是处理JSP页面中的标签的属性和内容,定义好之后就跟使用JSTL一样
<标签名 属性名="属性值" 属性名="属性值"> 标签的内容 <子标签名 属性名="属性值" 属性名="属性值"> 子标签的内容 </子标签名> </标签名>
经典标签处理器(JSP2.0以前):实现接口Tag、IterationTag、BodyTag的标签处理器都叫做经典标签处理器。
简单标签处理器(JSP2.0引入):实现接口SimpleTag或者继承类SimpleTagSupport的标签处理器叫做简单标签处理器。
// 自定义标签的相关接口都在 javax.servlet.jsp.tagext包中 public interface Tag extends JspTag public interface IterationTag extends Tag public interface BodyTag extends IterationTag public interface SimpleTag extends JspTag public class SimpleTagSupport extends java.lang.Object implements SimpleTag // 提供了接口Simpletag的所有方法的默认实现 public abstract class JspFragment extends java.lang.Object
接口simpleTag和支持类SimpleTagSupport中的方法
// SimpleTag中的方法 void doTag() throws JspException, java.io.IOException void setParent(JspTag parent) JspTag getParent() void setJspContext(JspContext pc) void setJspBody(JspFragment jspBody) // SimpleTagSupport中的方法 在SimpleTag基础上又增加了三个 protected JspFragment getJspBody() protected JspContext getJspContext() public static final JspTag findAncestorWithClass(JspTag from, java.lang.Class<?> klass)
使用自定义标签的过程如下
(1)浏览器访问JSP页面资源。
(2)Tomcat将JSP页面翻译成Servlet --> 编译该Servlet --> 构造类对象 --> 调用_jspService()方法。
(3)检查JSP文件的taglib指令,检查是否存在相应的tld文件,如果没有会报错。
(4)根据JSP页面中的标签名<easy:firstTag>在tld文件中找到相应的<tag>标签,并定位到标签处理器如下
<tag> <name>firstTag</name> <tag-class>app06a.MyFirstTag</tag-class> <body-content>empty</body-content> </tag>
(5)构造标签处理器app06a.MyFirstTag对象,然后调用标签处理器中的相应方法(将标签处理器的声明周期)
简单标签器的生命周期(即执行过程)如下: setJspContext(jspContext)和doTag()方法时一定会被JSP容器调用的
(1)JSP容器调用简单标签处理器的无参构造器并创建它的实例,因此简单标签处理器必须有无参构造器。
(2)JSP容器调用setJspContext(jspContext)方法,同时传入JspContext对象,该对象中最重要的方法是getOut()方法,其能够返回JspWriter对象,通过JspWriter就可以把相应返回前端了。通常情况下,要把传入的JspContext赋值给类的成员变量方便以后使用。
public void setJspContext(JspContext jspContext)
(3)如果标签处理器的定制标签嵌套再另一个自定义标签中,JSP容器就会调用setParent(jspTag)方法
public void setParent(JspTag parent)
(4)JSP容器调用该标签中所定义的每个属性的setter方法
(5)如果需要处理页面内容,JSP容器还会调用接口SimpleTag的setJspBody(jspBody)方法,把由JspFragment封装的页面内容传过来。如果没有页面内容则不会调用该方法。
public void setJspBody(JspFragment jspBody)
(6)JSP容器调用doTag()方法,所有变量在doTag方法返回时进行同步。doTag()方法只会执行一次,业务逻辑、遍历及页面内容操作都是在这里实现。
创建自定义标签的三步骤:
第一步:编写标签处理器,标签处理器就是一个实现了标签接口的Java类。在编写标签处理器时需要servlet-api.jar和jsp-api.jar这两个文件,不过Tomcat自带了,可以在Tomcat的lib目录下找到。 第二步:在标签库描述器中注册标签。标签库描述器是以.tld结尾的XML文件,该文件必须放在WEB-INF目录下(避免浏览器直接访问)。 第三步:使用自定义标签。需要用到<%@ taglib uri="uri" prefix="prefix" %>指令,其uri属性时标签库描述器的绝对路径或者相对路径,如果使用jar包中的标签库,就必须使用绝对路径了。
实例一
编写简单标签处理器
package app06a.customTag; import java.io.IOException; import javax.servlet.jsp.JspContext; import javax.servlet.jsp.JspException; import javax.servlet.jsp.JspWriter; import javax.servlet.jsp.tagext.JspFragment; import javax.servlet.jsp.tagext.JspTag; import javax.servlet.jsp.tagext.SimpleTag; public class MyFirstTag implements SimpleTag { // 简单标签处理器必须有无参构造器,此处使用默认的无参构造器 JspContext jspContext; @Override public void doTag() throws JspException, IOException { // JSP容器一定会调用该方法 System.out.println("doTag"); JspWriter out = jspContext.getOut(); out.println("This is my first tag."); } @Override public JspTag getParent() { System.out.println("getparent"); return null; } @Override public void setJspBody(JspFragment jspFragment) { // 如果需要处理页面内容,JSP容器会调用该方法,且在调用doTag()方法之前 System.out.println("setJspBody"); } @Override public void setJspContext(JspContext jspContext) { // JSP容器一定会调用该方法, 且首先调用该方法 this.jspContext = jspContext; System.out.println("setJspContext"); } @Override public void setParent(JspTag arg0) { // 如果自定义标签被另一个自定义标签所嵌套时,JSP容器调用该方法,且在调用doTag()方法之前 System.out.println("setParent"); } }
编写标签库处理文件
<?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> <description>Simple tag example</description> <tlib-version>1.0</tlib-version> <jsp-version>1.0</jsp-version> <short-name>My First taglib Example</short-name> <tag> <name>firstTag</name> <tag-class>app06a.customTag.MyFirstTag</tag-class> <body-content>empty</body-content> </tag> </taglib>
JSP页面中使用自定义标签
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="/WEB-INF/firstTag.tld" prefix="easy" %> <%-- 引入自定义标签描述库文件 --%> <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>Test my first tag</title> </head> <body> Hello <br /> <easy:firstTag></easy:firstTag> <!-- 使用自定义标签 --> </body> </html>
测试结果
实例二 处理属性 继承类SimpletagSupport
编写简单标签处理器
package app06a.customTag; import java.io.IOException; import java.util.StringTokenizer; import javax.servlet.jsp.JspContext; import javax.servlet.jsp.JspException; import javax.servlet.jsp.JspWriter; import javax.servlet.jsp.tagext.SimpleTagSupport; public class DataFormatterTag extends SimpleTagSupport { private String header; private String items; public void setHeader(String header) { //JSP容器调用,并且在调用doTag()之前 this.header = header; } public void setItems(String items) { this.items = items; } @Override public void doTag() throws IOException, JspException { // JSP容器调用, JspContext jspContext = getJspContext(); JspWriter out = jspContext.getOut(); out.println("<table style='border:1px solid green'>"); out.println("<tr><td><span style='font-weight:bold'>" + header + "</span></td><tr>"); StringTokenizer tonkenizer = new StringTokenizer(items, ","); while (tonkenizer.hasMoreTokens()) { String token = tonkenizer.nextToken(); out.println("<tr><td>" + token + "</td></tr>"); } out.println("</table>"); } }
编写标签库处理文件
<?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> <description>Simple tag example</description> <tlib-version>1.0</tlib-version> <jsp-version>1.0</jsp-version> <short-name>My First taglib Example</short-name> <tag> <name>firstTag</name> <tag-class>app06a.customTag.MyFirstTag</tag-class> <body-content>empty</body-content> </tag> <tag> <name>dataFormatter</name> <tag-class>app06a.customTag.DataFormatterTag</tag-class> <body-content>empty</body-content> <attribute> <!-- 属性 --> <name>header</name> <required>true</required> </attribute> <attribute> <name>items</name> <required>true</required> </attribute> </tag> </taglib>
JSP页面中使用自定义标签
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="/WEB-INF/firstTag.tld" prefix="easy" %> <%-- 指定标签库描述文件 --%> <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>Testing DataFormatterTag</title> </head> <body> <easy:dataFormatter header="States" items="Alabama,Alaska,Georgia,Florida" /> <!-- 该标签中有两个属性 --> <br /> <easy:dataFormatter header="Country" > <jsp:attribute name="items"> <!-- 使用JSP动作给标签中的属性赋值 --> IS,UK,Canada,Korea </jsp:attribute> </easy:dataFormatter> </body> </html>
测试
实例三 访问标签内容
在SimpleTag中,可以通过JSP容器传入的JspFragment对象来访问标签内容。JSP片段的定义不能包含脚本或者脚本表达式,只能是文本模板或者JSP标准节点。
JspFragment类中有两个方法
public abstract JspContext getJspContext() // 返回这个JspFragment关联的JspContext对象 public abstract void invoke(java.io.Writer out) throws JspException, java.io.IOException
// 执行标签的内容,然后通过指定的Writer对象将其直接输出,
// 如果把null传入invoke()方法中,那么这个Writer将会被JspFragment关联的JspContext对象中的getOut()方法返回的JspWriter对象所接管。
编写简单标签处理器
package app06a.customTag; import java.io.IOException; import javax.servlet.jsp.JspContext; import javax.servlet.jsp.JspException; import javax.servlet.jsp.JspWriter; import javax.servlet.jsp.tagext.JspFragment; import javax.servlet.jsp.tagext.SimpleTagSupport; public class SelectElementTag extends SimpleTagSupport { private String[] countries = {"Australia", "Brazil", "China", "Japan" }; @Override public void doTag() throws IOException, JspException { JspContext jspContext = getJspContext(); JspWriter out = jspContext.getOut(); JspFragment fragment = getJspBody(); out.println("<select>"); for (int i = 0; i < 4; i++) { jspContext.setAttribute("value", countries[i]); jspContext.setAttribute("text", countries[i]); fragment.invoke(null); } out.println("</select>"); } }
编写标签库处理文件
<?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> <description>Simple tag example</description> <tlib-version>1.0</tlib-version> <jsp-version>1.0</jsp-version> <short-name>My First Taglib Example</short-name> <tag> <name>select</name> <tag-class>app06a.customTag.SelectElementTag</tag-class> <body-content>scriptless</body-content> </tag> </taglib>
JSP页面中使用自定义标签
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="/WEB-INF/mytags.tld" prefix="easy" %> <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>Testing SelectElementFormatTag</title> </head> <body> <easy:select> <option value="${value }">${text }</option> <!-- 使用EL表达式 --> </easy:select> </body> </html>
测试结果
发布自定义标签
在tld文件中添加<uri>元素
将自定义的标签处理器和标签库描述文件打包到JAR包中
使用时,将JAR包放到应用的WEB/lib目录下,在使用的时候,任何使用自定义标签的JSP页面都要使用这个标签库描述器中定义的uri
可以将整个应用打包,使用时放到应用的WEB/lib目录下