JSP中的自定义标签
目录
简介
自定义标签主要用于移除JSP页面中的Java代码。
要使用自定义标签移除JSP页面中的Java代码,需要完成以下两个步骤:
- 编写一个实现Tag接口的Java类,并把页面中Java代码适当地转移到这个Java类中(标签处理类)。
- 编写标签库描述符(tld)文件(放置在WEB-INF目录下),在tld文件中把标签处理器类描述成一个标签。
入门案例
- 使用标签输出客户机IP
实现Tag接口的Java类 ViewIPTag
package com.wm103.web.tag;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.JspWriter;
import javax.servlet.jsp.tagext.TagSupport;
import java.io.IOException;
/**
* Created by DreamBoy on 2017/5/14.
*/
public class ViewIPTag extends TagSupport {
@Override
public int doStartTag() throws JspException {
HttpServletRequest request = (HttpServletRequest) this.pageContext.getRequest();
JspWriter out = this.pageContext.getOut();
String ip = request.getRemoteAddr();
try {
out.print(ip);
} catch (IOException e) {
throw new RuntimeException(e);
}
return super.doStartTag();
}
}
在WEB-INF目录下定义标签库描述符文件 wm103.tld
<?xml version="1.0" encoding="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 http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd"
version="2.0">
<description>A tag library exercising SimpleTag handlers.</description>
<tlib-version>1.0</tlib-version>
<short-name>WM103</short-name>
<uri>http://www.wm103.com</uri>
<tag>
<name>viewIP</name>
<tag-class>com.wm103.web.tag.ViewIPTag</tag-class>
<body-content>empty</body-content><!-- 标签体为空 -->
</tag>
</taglib>
在JSP页面中使用标签 index.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@ taglib uri="http://www.wm103.com" prefix="wm103"%>
<html>
<head>
<title>自定义标签</title>
</head>
<body>
<p>您的IP是:
<%
String ip = request.getRemoteAddr();
out.print(ip);
%>
</p>
<p>您的IP是:<wm103:viewIP/></p>
</body>
</html>
自定义标签功能扩展
自定义标签除了可以移除JSP页面Java代码外,它也可以实现以下功能:
1. 控制JSP页面某一部分内容是否执行;
2. 控制整个JSP页面是否执行;
3. 控制JSP页面内容重复执行;
4. 修改JSP页面内容输出。
传统自定义标签的运行原理
(JSP2.0以前的Tag接口)
1. 用户请求服务器JSP页面;
2. 服务器接收到请求,解析对应的JSP页面;
3. 解析过程中,JSP引擎遇到自定义标签,首先实例化标签对应的标签处理器类的实例对象,然后按照JSP规范定义的通信规则依次调用它的方法;
4. 调用标签处理器类的setPageContext
方法,把页面的pageContext
对象传递给标签处理器类;
5. 服务器判断该标签是否有父标签,如果存在父标签,则把父标签作为一个对象,调用标签处理器类的setParent
方法传递给标签处理器类;如果没有,则传递一个null
到方法中;
6. 完成以上标签的初始化工作后,服务器就开始执行标签,调用标签处理类的doStartTag
方法;
7. 如果标签有标签体,这时服务器一般会执行标签体;
8. 服务器遇到JSP页面结束标签,则调用标签处理器类的doEndTag
方法;
9. 整个标签执行完后,服务器一般情况会调用release()
方法释放标签工作时所占用的资源;(release
方法真正被调用的时候是在Web应用被销毁的时候,也就是第一次解析标签成功后,标签处理器类会驻留在内存中,为其他请求服务,直至停止Web应用时,Web容器才会调用release
方法)
10. 服务器接着再执行余下的JSP页面。
传统自定义标签的使用
注:以下案例是JSP2.0以前的实现方式!!
控制JSP页面部分内容执行
TagDemo1.java
package com.wm103.web.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.Tag;
import javax.servlet.jsp.tagext.TagSupport;
public class TagDemo1 extends TagSupport {
@Override
public int doStartTag() throws JspException {
return Tag.SKIP_BODY; // 不执行标签体
//return Tag.EVAL_BODY_INCLUDE; // 执行标签体
}
}
wm103.tld(向上述提到的wm103.tld文件增加新的tag标签,如无其他特别说明,下面案例也是这样的做法)
<tag>
<name>demo1</name>
<tag-class>com.wm103.web.tag.TagDemo1</tag-class>
<body-content>JSP</body-content><!-- 标签体为JSP内容 -->
</tag>
在JSP页面中的使用(记得先 <%@ taglib uri="http://www.wm103.com" prefix="wm103"%>
)
<wm103:demo1>
<p>使用标签控制页面内容是否输出</p>
</wm103:demo1>
控制整个JSP页面是否执行
TagDemo2.java
package com.wm103.web.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.Tag;
import javax.servlet.jsp.tagext.TagSupport;
/**
* Created by DreamBoy on 2017/5/14.
*/
public class TagDemo2 extends TagSupport {
@Override
public int doEndTag() throws JspException {
return Tag.SKIP_PAGE; // 不执行余下JSP页面
//return Tag.EVAL_PAGE; // 执行余下JSP页面
}
}
wm103.tld
<tag>
<name>demo2</name>
<tag-class>com.wm103.web.tag.TagDemo2</tag-class>
<body-content>empty</body-content>
</tag>
在JSP页面中的使用
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@ taglib uri="http://www.wm103.com" prefix="wm103"%>
<wm103:demo2/>
<html>
<head>
<title>用标签控制整个JSP是否输出</title>
</head>
<body>
<p>用标签控制整个JSP是否输出!!!</p>
</body>
</html>
控制标签体执行重复执行
TagDemo3.java
package com.wm103.web.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.IterationTag;
import javax.servlet.jsp.tagext.Tag;
import javax.servlet.jsp.tagext.TagSupport;
/**
* Created by DreamBoy on 2017/5/14.
* 控制标签体执行5次
*/
public class TagDemo3 extends TagSupport {
@Override
public int doStartTag() throws JspException {
return Tag.EVAL_BODY_INCLUDE; // 让它执行标签体
}
int x = 5;
@Override
public int doAfterBody() throws JspException {
x--;
if(x > 0) {
return IterationTag.EVAL_BODY_AGAIN;
} else {
x = 5; // 这句话不加的话,第二次用该标签时,就只会执行一次标签体。因为标签处理器类在第一次被使用时被实例化,再次使用会使用驻留在内存中的实例对象,x值仍为0。
return IterationTag.SKIP_BODY;
}
}
}
wm103.tld
<tag>
<name>demo3</name>
<tag-class>com.wm103.web.tag.TagDemo3</tag-class>
<body-content>JSP</body-content>
</tag>
在JSP页面中的使用
<p>
<wm103:demo3>
重复执行!
</wm103:demo3>
</p>
<p>
<wm103:demo3>
重复执行!
</wm103:demo3>
</p>
用标签修改JSP页面内容
TagDemo4.java
package com.wm103.web.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyContent;
import javax.servlet.jsp.tagext.BodyTag;
import javax.servlet.jsp.tagext.BodyTagSupport;
import java.io.IOException;
/**
* Created by DreamBoy on 2017/5/14.
* 修改标签体,把标签体内容修改为大写
*/
public class TagDemo4 extends BodyTagSupport {
@Override
public int doStartTag() throws JspException {
return BodyTag.EVAL_BODY_BUFFERED; // 让执行标签体时调用setBodyContent方法
}
@Override
public int doEndTag() throws JspException {
BodyContent bc = this.getBodyContent(); // 获得标签体
String content = bc.getString().toUpperCase();
try {
this.pageContext.getOut().write(content);
} catch (IOException e) {
throw new RuntimeException(e);
}
return super.doEndTag(); // return Tag.EVAL_PAGE;
}
}
wm103.tld
<tag>
<name>demo4</name>
<tag-class>com.wm103.web.tag.TagDemo4</tag-class>
<body-content>JSP</body-content>
</tag>
在JSP页面中的使用
<wm103:demo4>
DreamBoy
</wm103:demo4>
简单标签的运行原理
- 用户请求服务器JSP页面;
- 服务器接收到请求,解析对应的JSP页面;
- 解析过程中,遇到自定义标签,首先实例化标签对应的标签处理器类;
- 调用标签处理器类的
setJspContext
方法,把页面的pageContext
对象传递给标签处理器类; - 服务器判断该标签是否有父标签,如果存在父标签,则把父标签作为一个对象,调用标签处理器类的
setParent
方法传递给标签处理器类;如果没有,则传递一个null
到方法中; - 调用
setJspBody
方法,将标签体封装为JspFragment
实例对象,并传递给标签处理器类; - 执行页面中的自定义标签,执行标签实际上就是调用
doTag
方法; - 标签执行完成(这里跟传统自定义标签不同,被实例化的标签处理器对象不会驻留在内存中,而是使用完成后,由Java垃圾处理机制处理);
- 服务器接着再执行余下的JSP页面。
简单标签的使用
控制JSP页面部分内容执行
SimpleTagDemo1.java
package com.wm103.web.simpletag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.JspFragment;
import javax.servlet.jsp.tagext.SimpleTagSupport;
import java.io.IOException;
public class SimpleTagDemo1 extends SimpleTagSupport {
@Override
public void doTag() throws JspException, IOException {
// 不显示标签体的话就不调用如下语句
JspFragment jf = this.getJspBody();
jf.invoke(null); // 等价于 // jf.invoke(this.getJspContext().getOut());
}
}
simplewm103.tld
<?xml version="1.0" encoding="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 http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd"
version="2.0">
<description>A tag library exercising SimpleTag handlers.</description>
<tlib-version>1.0</tlib-version>
<short-name>WM103</short-name>
<uri>/SimpleWm103</uri>
<tag>
<name>demo1</name>
<tag-class>com.wm103.web.simpletag.SimpleTagDemo1</tag-class>
<body-content>scriptless</body-content>
</tag>
</taglib>
在JSP页面中的使用
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@ taglib uri="/SimpleWm103" prefix="swm103"%>
<html>
<head>
<title>使用简单标签控制页面内容(标签体)是否输出</title>
</head>
<body>
<swm103:demo1>
<p>使用简单标签控制页面内容是否输出</p>
</swm103:demo1>
</body>
</html>
控制整个JSP页面是否执行
SimpleTagDemo2.java
package com.wm103.web.simpletag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.SkipPageException;
import javax.servlet.jsp.tagext.SimpleTagSupport;
import java.io.IOException;
public class SimpleTagDemo2 extends SimpleTagSupport {
@Override
public void doTag() throws JspException, IOException {
throw new SkipPageException();
}
}
simplewm103.tld(在原来的simplewm103.tld中添加如下tag标签)
<tag>
<name>demo2</name>
<tag-class>com.wm103.web.simpletag.SimpleTagDemo2</tag-class>
<body-content>scriptless</body-content>
</tag>
在JSP页面中的使用
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@ taglib uri="/SimpleWm103" prefix="swm103"%>
<swm103:demo2/>
<html>
<head>
<title>用简单标签控制整个JSP是否输出</title>
</head>
<body>
<p>用简单标签控制整个JSP是否输出!!!</p>
</body>
</html>
控制标签体执行重复执行
SimpleTagDemo3.java
package com.wm103.web.simpletag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.JspFragment;
import javax.servlet.jsp.tagext.SimpleTagSupport;
import java.io.IOException;
public class SimpleTagDemo3 extends SimpleTagSupport {
@Override
public void doTag() throws JspException, IOException {
JspFragment jf = this.getJspBody();
for (int i = 0; i < 5; i++) {
jf.invoke(null);
}
}
}
simplewm103.tld(在原来的simplewm103.tld中添加如下tag标签)
<tag>
<name>demo3</name>
<tag-class>com.wm103.web.simpletag.SimpleTagDemo3</tag-class>
<body-content>scriptless</body-content>
</tag>
在JSP页面中的使用
<swm103:demo3>
重复执行!
</swm103:demo3>
用标签修改JSP页面内容
SimpleTagDemo4.java
package com.wm103.web.simpletag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.JspFragment;
import javax.servlet.jsp.tagext.SimpleTagSupport;
import java.io.IOException;
import java.io.StringWriter;
public class SimpleTagDemo4 extends SimpleTagSupport {
@Override
public void doTag() throws JspException, IOException {
JspFragment jf = this.getJspBody();
StringWriter writer = new StringWriter();
jf.invoke(writer);
String content = writer.getBuffer().toString().toUpperCase();
this.getJspContext().getOut().write(content);
}
}
simplewm103.tld(在原来的simplewm103.tld中添加如下tag标签)
<tag>
<name>demo4</name>
<tag-class>com.wm103.web.simpletag.SimpleTagDemo4</tag-class>
<body-content>scriptless</body-content>
</tag>
在JSP页面中的使用
<swm103:demo4>
DreamBoy
</swm103:demo4>
开发带属性的标签
自定义标签可以定义一个或多个属性,通过这些属性向标签处理器传递参数信息。
为自定义标签定义属性时,每个属性都必须按照JavaBean的属性命名方式,在标签处理器中定义属性名对应的setter方法,用来接收JSP页面调用自定义标签时传递进来的属性值。
要想让一个自定义标签具有属性,通常需要进行如下操作:
- 在标签处理器中编写每个属性对应的setter方法;
- 在TLD文件中描述标签的属性。
案例:
标签处理器类 SimpleTagDemo5.java
package com.wm103.web.simpletag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.JspFragment;
import javax.servlet.jsp.tagext.SimpleTagSupport;
import java.io.IOException;
import java.util.Date;
public class SimpleTagDemo5 extends SimpleTagSupport {
private int count;
private Date date;
public void setCount(int count) {
this.count = count;
}
public void setDate(Date date) {
this.date = date;
}
@Override
public void doTag() throws JspException, IOException {
this.getJspContext().getOut().write(this.date.toLocaleString() + "<br/>");
JspFragment jf = this.getJspBody();
for (int i = 0; i < count; i++) {
jf.invoke(null);
}
}
}
simplewm103.tld(在原来的simplewm103.tld中添加如下tag标签)
<tag>
<name>demo5</name>
<tag-class>com.wm103.web.simpletag.SimpleTagDemo5</tag-class>
<body-content>scriptless</body-content>
<attribute>
<name>count</name>
<required>true</required> <!--属性是否必须-->
<rtexprvalue>true</rtexprvalue> <!--设置为true后,count可以接收JSP脚本或EL表达式的值-->
</attribute>
<attribute>
<name>date</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
JSP页面
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@ page import="java.util.Date" %>
<%@ taglib uri="/SimpleWm103" prefix="swm103"%>
<html>
<head>
<title>开发带属性的标签</title>
</head>
<body>
<p>
<swm103:demo5 count="10" date="<%= new Date()%>">
<p>DreamBoy</p>
</swm103:demo5>
</p>
</body>
</html>