编写自定义标签

到不同的公司工作,都会遇到每个公司自己的标签,也即自定义的标签。其实,自定义标签主要用于移除Jsp页面中的java代码。

要实现自己自定义的标签,只需要完成以下两个步骤:

1.编写一个实现Tag接口的Java类(标签处理器类)

2.编写标签库描述符(tld)文件,在tld文件中对标签处理器类描述成一个标签

3.在jsp页面上引入再使用

在介绍自定义标签的开发时,先提前说下,对于自定义标签的开发有两种方式:

1.传统标签【了解】

步骤:

a.写一个类实现Tag接口

b.写一个tld文件,描述写好的类

c.在jsp页面中引入tld文件,然后就可以使用自定义的标签了。

在传统标签中,分为doStartTagdoEndTag方法来分别处理发现开始标签和发现结束标签时的代码,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文件:



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值