百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

Java教程:BodyTag接口用法详细介绍

yuyutoo 2024-10-12 01:06 5 浏览 0 评论

在实现自定义标签时,有时需要对标签体的内容进行处理以后再向浏览器输出,比如将小写英文字母转化为大写,将HTML标签进行转义等。为了实现这样的功能,JSP规范中它义了一个BodyTag接口,它继承自IterationTag接口,并在IterationTag接口基础上新增了两个方法和一个静态常量,具体如下。

1. EVAL_BODY_BUFFERED常量

如果标签处理器类实现了BodyTag接口,它的doStartTag()方法除了可以返回SKIP_BODY和EVAL_BODY_INCLUDE常量之外,还可以返回EVAL_BODY_BUFFERED常量。当doStartTag()方法返回EVAL_BODY_BUFFERED常量时,JSP容器将会创建个javax.servlet.jsp.tagext.BodyContent对象,使用该对象来执行标签体。关于BodyContent类的用法,将在下面进行详细的讲解。

2. setBodyContent(BodyContent b)方法

当且仅当doStartTag()方法返回EVAL_BODY_BUFFERED常量时,JSP容器才会调用setBodyContent()方法,通过该方法将BodyContent对象传递给标签处理器类使用。

3. dolnitBody()方法

JSP容器在调用setBodyContent()方法后会调用doInitBody()方法来完成些初始化工作,该方法的调用在标签体执行之前。

其中,最重要的是setBodyContent()方法。为了帮助大家更好地理解BodyTag接口处理标签内容的方式,有必要对BodyContent类进行详细讲解。

BodyContent类是JspWriter类的子类,它在JspWriter的基础上增加了一个用于存储数据的缓冲区(确切地说缓冲区是在BodyContent的子类org.apache.jasper.runtime.BodyContentImple中定义的),当调用BodyContent对象的方法写数据时,数据将被写人到BodyContent内部的缓冲区中。

明白了BodyContent类的这个特点,就不难理解JSP容器是如何利用BodyContent对象来处理标签体内容了。当标签处理器类的doStartTag()方法返回EVAL_BODY_BUFFERED常量时,JSP容器会创建一个BodyContent对象,然后调用该对象的write()方法将标签体的内容写人BodyContent对象的缓冲区中,开发者只要能够访问BodyContent缓冲区的内容,就能对标签体的内容进行处理。在BodyContent类中定义了一些用于访问缓冲区内容的方法,具体如下表所示。

BodyContent类的常用方法

方法声明

功能描述

String getString()

以字符串的形式返回BodyContent对象缓冲区中保存的数据

Reader getReader()

返回一个关联BodyContent对象缓冲区中数据的Reader对
象,通过Reader对象可以读取缓冲区中的数据

void clearBody()

用于清空BodyContent对象缓冲区中的内容

JspWriter getEnclosingWriter()

用于返回BodyContent对象中关联的JspWriter对象。当JSP容器创建BodyContent对象后,PageContext对象中的"out"属性不再指向JSP的隐式对象,而是指向新创建的BodyContent对象。同时,在BodyContent对象中会用一个JspWriter类型的成员变量enclosingWriter记住原来的隐式对象,getEnclosingWriter()方 法返回的就是原始的JSP隐式对象

writerOut(Writer out)

用于将BodyContent对象中的内容写人到指定的输出流

在上表列举的所有方法中,其中getEnclosingWriter()方法最难理解,但是,只需要记住该方法的返回值为out即可。

除了BodyContent类外,在BodyTag接口还会涉及很多常量和方法,为了让大家更好地掌握标签处理器的执行流程,接下来通过一张图来描述,具体如下图所示。

上图中清楚地描述了JSP容器执行标签处理器的过程。其中,release()方法之所以使用用虚线,是因为这个方法不会在标签处理器每次执行都被JSP容器调用,只有当标签处理器对象作为垃圾被回收之前它才会被调用。传统标签的处理器是单例的,只会被创建和销毁一次。

接下来,通过实现自定义标签,学习如何使用BodyTag接口将标签体中的小写英文字母转换为大写,具体步骤如下。

(1)编写标签处理器类ToUpperCase.java。

JSP规范中定义了一个类BodyTagSupport实现了BodyTag接口,为了简化程序的编写,标签处理器类ToUpperCase.java只需要继承BodyTagSupport类即可。ToUpperCase.java类的实现代码如下例所示。

package cn.itcast.chapter09.classisctag;
import java.io.IOException;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;
public class ToUpperCase extends BodyTagSupport {
    //定义doEndTag()方法
    public int doEndTag() throws JspException {
        //获取级冲区中数据
        String content = getBodyContent().getString();
        //将数据转为大写
        content = content.toUpperCase();
        try{
            //输出数据内容(两种方式均可)
            //pageContext.getout().write(content);
            bodyContent.getEnclosingWriter().write(content);
        } catch(IOException e){
            e.printStackTrace();
            return super.doEndTag();
        }
}

由于BodyTagSupport类中的doStartTag()方法默认返回EVAL_BODY_BUFFERED常量,JSP容器会在执行标签体之前创建BodyContent对象,然后将标签体内容通过setBodyContent()方法设置给BodyContent对象。因此在上面案例中的doEndTag()方法中可以直接使用getBodyContent()方法的getString()方法获得写人到BodyContent缓冲区中的内容,然后将其转换为大写,通过调用getEnclosingWriter()方法获取到out对象,将内容输出到浏览器中。

注意:不能直接使用doStartTag()方法的原因是,执行doStartTag()方法时,BodyContent对象中还没有缓存标签体的内容,因此通过getBodyContent()方法还无法获得标签的内容。

(2)注册标签处理器类。

在mytag.tld文件中增加一个Tag元素,对标签处理器类进行注册,注册信息如下所示。

<tag>
    <name>toUpperCase</name>
    <tag-class> cn.itcast.chapter09.classisctag.ToUpperCase</tag-class>
    <body-content>JSP</body-content>
</tag>

(3)编写JSP页面toUpperCase.jsp

在JSP页面中使用标签,在标签体中写人26个小写的英文字母,如下例所示。

<%@page language="java" pageEncoding="GBK"%>
<%@taglib uri= "http://www.itcast.cn" prefix="itcast"%>
<html>
    <head>
    <title>HelloWorld Tag</title>
</head>
<body>
    <itcast: toUpperCase>
        abcde fghij klmnopqrstuvwxyz
    </itcast:toUpperCase>
</body>
</html>

(4)启动Tomcat服务器,在浏览器地址栏中输人URL地址“http://localhost:8080/chapter09/toUpperCase.jsp”访问toUpperCase.jsp页面,从运行结果可以看出,自定义标签成功地将标签体中的小写英文字母转换为大写。

相关推荐

国内外注塑机及电脑密码大全(常见注塑机通用密码)

一、国外注塑机(日本、德国等)东洋注塑机万能码:9422345日精注塑机密码:222|7777DAMEG注塑机密码:000000000新泻注塑机密码:241650|261450住友注塑机密码:...

并发编程实战来咯(并发编程的艺术和并发编程实战)

提到并发编程,就不得不提C++ConcurrencyinAction(SecondEdition)(《C++并发编程实战第2版》)啦!《C++并发编程实战第2版》英文原版&中文译版看到这个...

无锁队列Disruptor原理解析(无锁队列应用场景)

队列比较队列...

理解 Memory barrier(内存屏障)(内存屏障 volatile)

...

并发编程 --- CAS原子操作(cas概念、原子类实现原理)

...

无锁CAS(附无锁队列的实现)(cas是一种无锁算法)

本文所有代码对应的Github链接为:https://github.com/dongyusheng/csdn-code/tree/master/cas_queue...

Linux高性能服务器设计(linux 服务器性能)

C10K和C10M计算机领域的很多技术都是需求推动的,上世纪90年代,由于互联网的飞速发展,网络服务器无法支撑快速增长的用户规模。1999年,DanKegel提出了著名的C10问题:一台服务器上同时...

浅谈Go语言的并发控制(go语言 并发)

前言本文原创,著作权归...

Datenlord |Etcd 客户端缓存实践(etcd 数据存储)

简介和背景...

无锁编程——从CPU缓存一致性讲到内存模型

缓存是一个非常常用的工程优化手段,其核心在于提升数据访问的效率。缓存思想基于局部性原理,这个原理包括时间局部性和空间局部性两部分:...

打通 JAVA 与内核系列之 一 ReentrantLock 锁的实现原理

...

如何利用CAS技术实现无锁队列(cas会锁总线吗)

linux服务器开发相关视频解析:...

Kotlin协程之一文看懂Channel管道

概述Channel类似于Java的BlockingQueue阻塞队列,不同之处在于Channel提供了挂起的send()和receive()方法。另外,通道Channel可以...

详解C++高性能无锁队列的原理与实现

1.无锁队列原理1.1.队列操作模型...

Javascript 多线程编程的前世今生

...

取消回复欢迎 发表评论: