到不同的公司工作,都会遇到每个公司自己的标签,也即自定义的标签。其实,自定义标签主要用于移除Jsp页面中的java代码。
要实现自己自定义的标签,只需要完成以下两个步骤:
1.编写一个实现Tag接口的Java类(标签处理器类)
2.编写标签库描述符(tld)文件,在tld文件中对标签处理器类描述成一个标签
3.在jsp页面上引入再使用
在介绍自定义标签的开发时,先提前说下,对于自定义标签的开发有两种方式:
1.传统标签【了解】
步骤:
a.写一个类实现Tag接口
b.写一个tld文件,描述写好的类
c.在jsp页面中引入tld文件,然后就可以使用自定义的标签了。
在传统标签中,分为doStartTag和doEndTag方法来分别处理发现开始标签和发现结束标签时的代码,在doStartTag可以通过返回值来控制标签体是否允许执行,
在doEndTag方法里可以通过返回值控制标签之后的剩余页面是否允许执行。
传统标签的这种开发方式,需要我们分析发现开始标签和发现结束标签时都需要执行什么代码,还需要分析到底要返回什么样的标签体控制程序执行,相对来说相当的 繁琐。
Tag接口的执行流程为:
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 方法。
下面就举一个简单的例子来说明一下,使用传统标签自定义标签:
首先编写一个类实现Tag接口
package com.jjyy.tag; import java.io.IOException; import javax.servlet.jsp.JspException; import javax.servlet.jsp.PageContext; import javax.servlet.jsp.tagext.Tag; /** * 传统标签 * @author JiangYu * */ public class PrintIp implements Tag { private PageContext pc = null; public int doEndTag() throws JspException { return 0; } public int doStartTag() throws JspException { String ip = pc.getRequest().getRemoteAddr(); try { pc.getOut().write(ip); } catch (IOException e) { e.printStackTrace(); } return 0; } public Tag getParent() { return null; } public void release() { } public void setPageContext(PageContext pc) { this.pc = pc; } public void setParent(Tag t) { } }然后,编写一个tld文件,放在WEB-INF目录下,如图
其中内容为:
<?xml version="1.0" encoding="UTF-8"?> <taglib version="2.0" 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"> <tlib-version>1.0</tlib-version> <short-name>ip</short-name> <uri>http://www.jjyy.com/MyTag</uri> <tag> <name>showIp</name> <tag-class>com.jjyy.tag.PrintIp</tag-class> <body-content>empty</body-content> </tag> </taglib>
最后,编写一个jsp页面,在其中引入自定义的标签,在页面上使用:
<%@ page language="java" contentType="text/html; charset=utf-8" pageEncoding="utf-8"%> <%@ taglib uri="http://www.jjyy.com/MyTag" prefix="ip" %> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8"> <title>Insert title here</title> </head> <body> 传统标签输出IP:<ip:showIp/> <br> </body> </html>
到此,实现自定义的标签过程就是上面三步了。
2.简单标签
上面简单的介绍了一下传统的标签,由于传统标签使用三个标签接口来完成不同的功能,显得过于繁琐,不利于标签技术的推广, SUN公司为降低标签技术的学习难 度,在JSP 2.0中定义了一个更为简单、便于编写和调用的SimpleTag接口来实现标签的功能。
我们可以看下,Tag接口的层次机构和里面的方法:
实现SimpleTag接口的标签通常称为简单标签。下面也介绍一下简单标签的使用
(1)写一个类实现SimpleTag接口(继承SimpleTag接口的默认实现类SimpleTagSupport)
(2)写一个tld文件,描述写好的类
(3)在jsp页面中引入tld文件,就可以在jsp页面中使用自定义标签了
SimpleTag中有五个方法:分别介绍下
1.setJspContext方法----用于把JSP页面的pageContext对象传递给标签处理器对象
2.setParent方法----用于把父标签处理器对象传递给当前标签处理器对象
3.getParent方法----用于获得当前标签的父标签处理器对象
4.setJspBody方法----用于把代表标签体的JspFragment对象传递给标签处理器对象
5.doTag方法----用于完成所有的标签逻辑,包括输出、迭代、修改标签体内容等。在doTag方法中可以抛出javax.servlet.jsp.SkipPageException异常,用
于通知WEB容器不再执行JSP页面中位于结束标记后面的内容,这等效于在传统标签的doEndTag方法中返回Tag.SKIP_PAGE常量的情况。
简单标签的原理
1.当jsp在执行的过程中,每当遇到一个简单标签时都会创建一个处理类对象.
2.调用setJspContext传入当前jsp页面的PageContext对象.
3.如果当前标签有父标签则调用setParent方法将父标签传入,如果没有父标签则这个方法不会被调用.
4.如果该标签具有属性,调用属性的setXXX方法将属性的值传入
5.如果当前标签具有标签体,则会调用setJspBody将封装了标签体信息的JspFragment传入,如果没有标签体,这个方法不执行
6.最后调用doTag方法,在这个方法里我们可以书写处理标签事件的java代码
7.当自定义标签执行完成后,简单标签对象就销毁掉了.
下面就写一个Demo来操作一下简单标签的开发过程:
首先还是一样,编写一个类实现simpleTag,其中有五个方法,但是sun为了避免我们写过多的代码,所以提供了一个SimpleTagSupport类,其实现了SimpleTag接
口,所以我们直接可以继承SimpleTagSupport类,重写doTag方法就可以了。
package com.jjyy.simpletag; import java.io.IOException; import java.io.StringWriter; import javax.servlet.jsp.JspException; import javax.servlet.jsp.SkipPageException; import javax.servlet.jsp.tagext.JspFragment; import javax.servlet.jsp.tagext.SimpleTagSupport; /** * 简单自定义标签 * @author JiangYu */ public class SimpleTag01 extends SimpleTagSupport { @Override public void doTag() throws JspException, IOException { } }然后编写tld文件:
<?xml version="1.0" encoding="UTF-8"?> <taglib version="2.0" 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"> <tlib-version>1.0</tlib-version> <short-name>ip</short-name> <uri>http://www.jjyy.com/MyTag</uri> <tag> <name>simpleTag1</name> <tag-class>com.jjyy.simpletag.SimpleTag01</tag-class> <body-content>scriptless</body-content> <attribute> <name>times</name> <required>true</required> <rtexprvalue>true</rtexprvalue> <type>int</type> </attribute> </tag> </taglib>最后在jsp页面中引入:
<%@ page language="java" contentType="text/html; charset=utf-8" pageEncoding="utf-8"%> <%@ taglib uri="http://www.jjyy.com/MyTag" prefix="ip" %> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8"> <title>Insert title here</title> </head> <body> 标签之前 <ip:simpleTag1>标签体</ip:simpleTag1> 标签之后 </body> </html>
要实现自己想要的功能我们只需要修改doTag方法就可以了。期望实现的功能:
1.控制标签体是否执行
package com.jjyy.simpletag; import java.io.IOException; import javax.servlet.jsp.JspException; import javax.servlet.jsp.tagext.JspFragment; import javax.servlet.jsp.tagext.SimpleTagSupport; /** * 简单自定义标签 * @author JiangYu */ public class SimpleTag01 extends SimpleTagSupport { @Override public void doTag() throws JspException, IOException { //1.控制标签体是否执行 //控制标签体执行--只要调用封装着标签体的JSPFragment对象的invoke()方法即可 JspFragment fragment = super.getJspBody(); //fragment.invoke(super.getJspContext().getOut()); fragment.invoke(null); } }
2.控制标签之后的内容是否执行
package com.jjyy.simpletag; import java.io.IOException; import javax.servlet.jsp.JspException; import javax.servlet.jsp.SkipPageException; import javax.servlet.jsp.tagext.SimpleTagSupport; /** * 简单自定义标签 * @author JiangYu */ public class SimpleTag01 extends SimpleTagSupport { @Override public void doTag() throws JspException, IOException { //2.控制标签之后的内容是否执行 //控制标签之后的内容执行:什么都不做,标签之后的内容就会执行 //控制标签之后的内容不执行:抛出SkipPageException这样的一个异常就可以控制标签之后的内容不执行 throw new SkipPageException(); } }
3.控制标签体重复执行
package com.jjyy.simpletag; import java.io.IOException; import javax.servlet.jsp.JspException; import javax.servlet.jsp.tagext.JspFragment; import javax.servlet.jsp.tagext.SimpleTagSupport; /** * 简单自定义标签 * @author JiangYu */ public class SimpleTag01 extends SimpleTagSupport { private int times;//对应tld文件中的attribute @Override public void doTag() throws JspException, IOException { //3.控制标签体重复执行 JspFragment fragment = super.getJspBody(); for(int i=0;i<times;i++){ fragment.invoke(null); } } public void setTimes(int times) { this.times = times; } }修改jsp页面
<%@ page language="java" contentType="text/html; charset=utf-8" pageEncoding="utf-8"%> <%@ taglib uri="http://www.jjyy.com/MyTag" prefix="ip" %> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8"> <title>Insert title here</title> </head> <body> <!-- 标签之前 <ip:simpleTag1 times="6">标签体</ip:simpleTag1> 标签之后 --> <ip:simpleTag1 times="6">jjyy</ip:simpleTag1> </body> </html>
4.修改标签体后进行输出
package com.jjyy.simpletag; import java.io.IOException; import java.io.StringWriter; import javax.servlet.jsp.JspException; import javax.servlet.jsp.tagext.JspFragment; import javax.servlet.jsp.tagext.SimpleTagSupport; /** * 简单自定义标签 * @author JiangYu */ public class SimpleTag01 extends SimpleTagSupport { @Override public void doTag() throws JspException, IOException { //4.修改标签体后进行输出 StringWriter sw = new StringWriter(); JspFragment fragment = super.getJspBody(); fragment.invoke(sw); String str = sw.toString(); str = "jjyy 测试自定义标签!!"; getJspContext().getOut().write(str); } }
通过上面的演示,基本上我们也可以开发自己定义的标签了。最后还是分析一下tld文件: