
JSP Tag 文件在简化 JSP 开发方面发挥着重要作用,它允许开发者将常用的页面逻辑封装成可重用的组件。然而,当需要在 Tag 的 Body 中包含完整的 JSP 代码,例如 Scriptlets (<% ... %>) 和表达式 (<%= ... %>) 时,Tag 文件便会遇到限制。
JSP Tag 文件通过 <%@tag %> 指令定义,并使用 body-content 属性来指定 Body 内容的类型。常用的 body-content 值包括 empty、scriptless 和 tagdependent。
- empty: 表示 Tag 没有 Body。
- scriptless: 表示 Body 中只允许 HTML 和 EL 表达式。
- tagdependent: 表示 Body 内容由 Tag 自身处理,JSP 引擎不会对其进行评估。
问题:Tag 文件无法支持完整的 JSP 代码
如果尝试在 body-content 设置为 scriptless 的 Tag 文件的 Body 中使用 Scriptlets 或表达式,将会遇到类似以下错误信息:
Scripting elements (
这意味着,JSP Tag 文件无法像 Java 类实现的 BodyTag 那样,通过 EVAL_BODY_INCLUDE 来评估 Body 中包含的完整 JSP 代码。
解决方案:使用 Java 类实现 BodyTag 接口
为了克服 JSP Tag 文件的限制,可以使用 Java 类来实现 BodyTag 接口。BodyTag 接口提供了对 Tag Body 内容进行完全控制的能力,包括评估和修改。
以下是一个示例,展示如何使用 Java 类来实现一个 Dialog Tag,它可以将 Body 内容包装在一个 div 元素中:
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.JspWriter;
import javax.servlet.jsp.PageContext;
import javax.servlet.jsp.tagext.BodyContent;
import javax.servlet.jsp.tagext.BodyTag;
import javax.servlet.jsp.tagext.Tag;
public class Dialog implements BodyTag {
private PageContext pageContext;
private BodyContent bodyContent;
@Override
public void setPageContext(PageContext pageContext) {
this.pageContext = pageContext;
}
@Override
public void setParent(Tag tag) {
// 不需要父标签
}
@Override
public Tag getParent() {
return null;
}
@Override
public void setBodyContent(BodyContent bodyContent) {
this.bodyContent = bodyContent;
}
@Override
public int doStartTag() throws JspException {
try {
JspWriter out = pageContext.getOut();
out.print("<div class=\"dialog\">");
} catch (Exception e) {
throw new JspException("Error in doStartTag", e);
}
return EVAL_BODY_BUFFERED; // 重要:使用 EVAL_BODY_BUFFERED 才能获取 BodyContent
}
@Override
public int doEndTag() throws JspException {
try {
JspWriter out = pageContext.getOut();
if (bodyContent != null) {
out.print(bodyContent.getString()); // 输出 Body 内容
}
out.print("</div>");
} catch (Exception e) {
throw new JspException("Error in doEndTag", e);
}
return EVAL_PAGE;
}
@Override
public void release() {
// 释放资源
}
@Override
public int doAfterBody() throws JspException {
return SKIP_BODY; // 只评估一次 Body
}
}使用示例:
-
编写 Tag Library Descriptor (TLD) 文件: 创建一个 TLD 文件 (例如 mytaglib.tld),并在其中声明 Dialog Tag。
<?xml version="1.0" encoding="UTF-8"?> <taglib xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" version="3.0"> <tlib-version>1.0</tlib-version> <short-name>mytaglib</short-name> <uri>/WEB-INF/tlds/mytaglib</uri> <tag> <name>dialog</name> <tag-class>com.example.Dialog</tag-class> <!-- 替换为你的类路径 --> <body-content>JSP</body-content> </tag> </taglib> -
在 JSP 页面中使用 Tag: 在 JSP 页面中引入 Tag 库,并使用 Dialog Tag。
<%@ taglib prefix="mytaglib" uri="/WEB-INF/tlds/mytaglib" %> <mytaglib:dialog> <%-- 这里可以使用完整的 JSP 代码 --%> <h1>Hello World!</h1> <p>This is a paragraph inside the dialog.</p> <% int x = 10; %> <%= x * 2 %> </mytaglib:dialog>
注意事项:
- 确保将编译后的 Java 类 (例如 Dialog.class) 放置在 Web 应用的 WEB-INF/classes 目录下。
- 确保 TLD 文件放置在 WEB-INF/tlds 目录下。
- 使用 EVAL_BODY_BUFFERED 在 doStartTag 方法中,这样才能获取 BodyContent。
- 使用 bodyContent.getString() 在 doEndTag 方法中输出 Body 内容。
- body-content 在 TLD 文件中声明为 JSP,表示允许使用完整的 JSP 代码。
总结:
JSP Tag 文件在某些场景下非常有用,但它在处理包含完整 JSP 代码的 Body 内容时存在限制。如果需要在 Tag 的 Body 中使用 Scriptlets 和表达式,必须采用 Java 类实现 BodyTag 接口的方式。通过 BodyTag 接口,可以完全控制 Tag Body 内容的评估和处理,从而实现更灵活和强大的 Tag 组件。虽然使用 Java 类实现 Tag 相对复杂,但它提供了更大的灵活性和控制力,尤其是在需要处理复杂的 Body 内容时。










