标准的JSP 标记可以调用JavaBeans组件或者执行客户的请求,这大大降低了JSP开发的复杂度和维护量。JSP技术也允许你自定义taglib,其实换句话说,taglib可以看成是对JSP标记的一种扩展,正如xml是对html的一种扩展一样。taglib通常定义在tag标签库中,这种标签库存放着你自己定义的tag标签。简而言之,如果使用taglib,那么你可以设计自己的JSP标记!
` lpz-"EEV t;q7t!sC] 一般来说,自定义tag标签主要用于操作隐藏对象、处理html提交表单、访问数据库或其它企业级的服务,诸如邮件和目录操作等等。自定义tag标签的使用者一般都是那些对java编程语言非常精通,而且对数据访问和企业级服务访问都非常熟悉的程序员,对于HTML设计者来说,使得他可以不去关注那些较复杂的商业逻辑,而将精力放在网页设计上。同时,它也将库开发者和库使用者进行合理分工,自定义tag标签将那些重复工作进行封装,从而大大提高了生产力,而且可以使得tag库可用于不同的项目中,完美地体现了软件复用的思想。
)}lRd#V vT{ kL 在这篇文章中,我们主要讨论:
J%rP$O$ dJuD|9R · 什么是自定义tag标签?
|zsbW9
W*m 9";sMB}W* · 怎么使用tag标签?
qYB~VE03 PS>x,T o 声明要使用的tag库
tjnPyaJEl S;\R!%t_ o 找到与之对应的tag处理类
d j5hv~ hlbvt-C?}" o tag标签的类型
)0 Z! n x_w~G]! / · 自定义tag标签
.T B"eUy c-1q2y o tag处理类
N3A<:%s YlK7;yrq( o tag库描述
7L(eh7 OYL]j{ o tag标签示例
S/A1RUt n{5NNV6 o 带属性的tag
:py\| ~f>2U]F>5 o 带body的tag
np$zo ;/O#4]2* o 定义了脚本变量的tag
+b$S~0n
Gpj* V|J o 具有协作关系的tag
S]/b\B.h+ E9fxjI%1 · 自定义tag标签
t;qP']2
f i~I@KJ> o 一个迭代tag的例子
},-* A$/\1282 o 一个模板tag库
W#F Q,+0) r`'y?Bra; o tag处理类到底是怎样被调用的?
"9y(
} <E,%@ ,T{<vRj7_ 什么是自定义的tag?
wVl+]zB ,?cH"@RJ 一个自定义的tag标签是用户定义的一种JSP标记。当一个含有自定义的tag标签的JSP页面被jsp引擎编译成servlet时,tag标签被转化成了对一个称为tag处理类的对象进行的操作。于是当JSP页面被jsp引擎转化为servlet后,实际上tag标签被转化成为了对tag处理类的操作。
}%!tT\8 EM j;2! 自定义tag标签有很多特色,诸如:
u0s8yPA fRZ KEIyk · 可以在JSP页面中自定义tag标签的属性
#E7AmmqD% MHj,<|8Q · 访问JSP页面中的所有对象
f{#j6wZM |9K<-yD · 可以动态地修改页面输出
1`bl&}6l|E -1|iz2^N · 彼此这间可以相互通信。你可以先创建一个JavaBeans组件,然后在一个tag中调用此JavaBeans组件,同时可以在另一个tag中调用它。
A#h /B+ T Eu'*>g · tag允许相互嵌套,可以在一个JSP页面中完成一些复杂的交互。
;T +pu>) (<*e .N m su+s 使用tag标签
=<z.mzqu5 ;az5ZsvN
D l1=JrpCan 本节主要描述怎样在JSP页面中使用tag标签,以及tag标签的不同类型。
K{ fsn4rk LaMLv<)k 要使用tag标签,JSP程序员必须做2件事:
"B7`'jz V7Yaks · 声明此tag标签的tag库
&}6KPA; XL}"1lE · 实现此tag标签
C(T;>if0NH 25y6a|` 声明tag标签所在的tag库
/'.=sH 2;3f=$3 如果要使用tag标签,则应用JSP的taglib指示符来指定其tag库(注意:taglib要在在使用此tag标签之前声明)
Y4.Eq+$gh 0u
B'g+MU` <%@ taglib uri=”/WEB-INF/tutorial-template.tld” prefix=”tt” %>
xw^.bz| a`Qot uri属性定义了唯一的标签库描述(以下简称TLD),它可以是直接是tld文件名或一个独一无二的名字。prefix是用来区别其它TLD中和本TLD中有重名的tag的一种手段。
m}
?rJ Ey[On^$ TLD必须以.tld作为扩展名,并且存放在当前应用的WEB-INF目录或其子目录下。你可以通过它的文件名直接引用它,也可以通过别的方式间接地引用它。
AV8T a,t``'c; 以下taglib指示符直接引用一个TLD:
t(!r8!c
u} _6@hTen` <%@ taglib uri=”/WEB-INF/tutorial-template.tld” prefix=”tt” %>
58xnB!h\} u6u=2 以下的taglib指示符通过一个逻辑名称间接地引用一个TLD:
YcX/{L[9o
kzmQm <%@ taglib uri=”/tutorial-template” prefix=”tt” %>
"6ECgyD+E! B$Z3+$hfF 如果是间接引用TLD的话,那你必须还要在web.xml中定义此逻辑名称与tld文件之间的映射,具体做法是在web.xml中加入一个名为taglib的元素:
BQ</g* $; i+3fhV <taglib>
penlG36Q ,|?CU
r9Y <taglib-uri>/tutorial-template</taglib-uri>
t`'iU$:1f /Dtd#OAdr <taglib-location>
[P4$Khu$ QYb33pN| /WEB-INF/tutorial-template.tld
t+r:"bb ?']h%'Q
</taglib-location>
&hIRd,1# 7IlOG~DC </taglib>
wd@aw / j9+I0>#X FXdD4 X) 实现此tag标签
:p&!RI(l g.JN_t5 3VnQnd E 为了实现tag标签,你有2种方法来存放tag处理类。一、让tag处理类以.class的方式存放于当前应用的WEB-INF/class子目录下,二、如果tag处理类是以JAR包的形式存在的话,那可以放在当前应用的WEB-INF/lib目录下,如果tag处理类要在多个应用中共享,那么它就应放在jsp服务器上的common/lib目录下,对于tomcat来说,就是tomcat/common/lib目录下。
-~" :f8 d@0Kr5_ 5v\!]?(O; tag标签类型
,B<l @Y,7'0U ^-CINt{O 自定义的tag标签遵循XML语法。它有一个开始标记和一个结束标记,有的还有body(即文本节点):
sd#|3 _/* U2.xS <tt:tag>
2 !"
XzdD `~UZU@/x body
I:V0Xxz5t 1{Mcs%W;w5 </tt:tag>
D<i[LZd u}bf-;R y;?ie]3G 一个不带body的tag标签如下:
+y2[msBs 3=Ec" <tt:tag />
;8S/6FI ,fIe&zq l%z< (L5 简单的tag标签
:4S%'d7 d1@%W;qX! 一个没有body和属性的tag标签如下:
FOwDp0 )Rat0$6 <tt:simple />
=$8nUX` P #F=c34u p,pR!qC> 带属性的tag标签
)? M9|u g[>\4B9t }0`nvAf 自定义标签可以有自己的属性。属性一般在开始标记中定义,语法为 attr=”value”。属性的作用相当于自定义标签的一个参数,它影响着tag处理类的行为。你可以在TLD中详细定义它。
,B&fFis WnwhSr2 你可以用一个String常量给一个属性赋值,也可以通过表达式给它赋值,如<%= ...%>。以struts为例,它的logic:present标签就是用的String常量来给属性赋值:
>R{qESmP= @pI5lh <loglic:present parameter = “Clear”>
!|q<E0@w\ zOEY6lAwI 而另一个标签logic:iterate是用表达式来给属性赋值:
^ 5VK> Fhz*&JC# <logci:iterate collection=”<%= bookDB.getBooks() %>”
]{
BEr* *tOG*hwdT id=”book” type=”database.BookDetails”>
8E&XbqP+ uV-'~8 _ ~RpGX 带body的tag标签
w:Jrmx 22<0DhJ 一个自定义标签可以包含其它自定义标签、脚本变量、HTML标记或其它内容。
m+{K^kr[ cWGDee( 在下述例子中,此JSP页面使用了struts的logic:present标签,如果些标签定义了parameter=”Clear”的属性,则将清除购物车的内容,然后打印出一条信息:
b5IA"w 5WqXo{S <logic:present parameter=”Clear”>
VN0mDh?E 1Kvx1p
<% cart.clear(); %>
}7G8|54t p2J|Hl| <font color=”#ff0000” size=”+2”><strong>
dt[k\ !-v L{l6Dd43q 你选择了清除购物车!
aw?=hXR! }~h'FHCC+ </strong></font>
<X TU8G \ 6EKgC1 </logic:present>
{
74mf'IW vhvdKD
{:c]|^w6 到底是用属性还是用body来传递信息?
zL5d0_E9 FVv8-- 如上所述,我们既可以通过属性,也可以通过body来传递信息。但一般来说,比较简单的类型,如字符串或简单表达式最好采用属性来传递信息。
>U1R.B7f sC00un%
O=) 定义脚本变量的tag标签
_8}QlT <]C$xp<2 所谓脚本变量,是指JSP中可以调用的变量或对象。它可由tag标签产生。以下示例阐述了一个tag标签定义了一个名为tx的由JNDI所定义的事务处理对象。脚本变量可以是ejb对象、事务、数据库连接等等:
k"kJ_( )CI1; <tt:lookup id=”tx” type=”UserTransaction” name=”java:comp/UserTransaction” />
o ]Jv;Iy@? 4RKW <% tx.begin(); %>
*F>v]8 30PZ{c&Rll ...
&Gm$:T'~ #B'aU#$u PCd0 ?c 具有协作关系的tag标签
p
<eC<dtu 41#w|L
\ 自定义tag标签之间可以通过共享对象来实现协作。在下述例子中,标签tag1创建了一个名为obj1的对象,在标签tag2仍可以重复使用obj。
=H0vE7 {* 1%M&CX <tt:tag1 attr1=”obj1” value1=”value” />
UV}73Sp $(s\{(Wn <tt:tag2 attr1=”obj1” />
sDgXU@ d;Hn#2C 在以下这个例子当中,如果外层的tag标签创建了一个对象,那么其内层的所有tag标签都可以使用这个对象。由于这样产生的对象没有一个指定的名字,那么就可以将少重名的冲突。这个例子阐述了一系列协作的嵌套对象。
:ztr) :"6q,W <tt:outerTag>
rQ4*k'lA: sUl6hX4 <tt:innerTag />
%CZ-r"A 7;.xc{ </tt:outerTag>
>}~#>Ru 0Q"u#V Sp }14{2=!Q Tag处理类
U(&oj e
N-lGa@ j EG'[`<*h Tag处理类必须实现Tag接口或BodyTag接口,不过现在一般都流行从TagSupport或BodyTagSupport类中继承,这些类或接口都可以在javax.servlet.jsp.tagext包中找到。
"HD+rmUEH
qLncn}oNM 当JSP引擎看到自己的JSP页面中包含有tag标签时,它会调用doStartTag方法来处理tag标签的开头,调用doEndTag方法来处理tag标签的结束。
h9$ Fx PIA)d-Z 下表说明不同类型的tag所需要不同的处理过程:
O#nR>1h ?3z- _8# Tag处理类的方法
fsO9EEn7X Jxf~&!zR Tag标签类型
8T;IZ(s 所调用的方法
KDi|( 5q{h 2).) 基本标签
2Zuq?1= doStartTag, doEndTag, release
c_{z(W" +c:3o* 带属性的标签
@Un/c:n doStartTag, doEndTag, set/getAttribute1...N, release
+&tgJ07A k.h`Cji@ 带内容的标签
w&Dv8Wv+Oq doStartTag, doEndTag, release
"wH) mQnd M|T4~Q U& 带内容的标签,且内容重复循环
T1B|w"In doStartTag, doAfterBody, doEndTag, release
e"-X U@`k1 +y[@T6_ 带内容的标签,且内容与JSP交互
O9v_y+M+M doStartTag, doEndTag, release, doInitBody, doAfterBody, release
LCXO>MXN )g|
BMmB 一个tag处理类可以通过javax.servlet.jsp.PageContext来与JSP交互,通过javax.servlet.jsp.PageContext类,tag处理类可以访问JSP中的request、session和application对像。
>-T`0wI di9!lS$ 如果tag标签是互相嵌套的,那内层的tag处理类可以通过它的parent属性来访问上层的tag处理类。
.=9s1~] -0o[f53}p 一般情况都将所有的tag处理类打成了JAR的包,以便于发布。
q^8EOAvnZ mIZwAKo C`oa3B,z Tag库描述(简称TLD)
oC*ees
g_ %kf>&b,Mi ~>G]_H]? Tag库是用xml语言描述的,TLD包括了tag库中所有tag标签的描述,它一般用来被jsp服务器用来校验tag的语法正确性,或者被jsp开发者用来开发新的标签。
WV;=@v O(2cWQ TLD的文件扩展名必须为.tld,而且必须放在当前WEB应用的WEB-INF目录或其子目录中。
W:&R~R NX* O_/ 一个TLD的内容的开头必须遵守标准的XML开头,用于描述DTD和xml的版本,例如:
t$&Qv) qOCJT Og7 <?xml version="1.0" encoding="ISO-8859-1" ?>
1F[L"W;r OL59e%X <!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN" "
http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd">
*#>F.#9 TLD必须以<taglib>来作为它的根元素,<taglib>的子元素如下表:
g]mtFrP {B$2"q/~ Ftb%{[0}u3 <taglib>的子元素
wn+FTqj XR# ;{p+b Element
x
FJg Description
.r|*Ch#;P ]rd/;kg.S tlib-version
7D" %%|:
h Tag库的版本
S^RUw W-2i+g) jsp-version
4\X||5.c Tag库所需要的jsp的版本
~d>%,?zz Z%o7f6P0IX short-name
={(j`VSUX0 助记符,tag的一个别名(可选)
I\P Bu$Ww @B1{r|-<^ uri
^~ =9 用于确定一个唯一的tag库
b=##A OTvROJP display-name
Ry;$^.7% 被可视化工具(诸如Jbuilder)用来显示的名称(可选)
hAR?
t5c ,+/zH'U} small-icon
#5CI)4x0! 被可视化工具(诸如Jbuilder)用来显示的小图标(可选)
I8m:3fL" #mc!Wt10 large-icon
J07O:cjyu 被可视化工具(诸如Jbuilder)用来显示的大图标(可选)
3*S[eqMJc uF(k[[qaiN description
.:1qK<vz 对tag库的描述(可选)
I@7/jUO 0LVE@qEL listener
.CV _\ 参见下面listener元素
`"y`AY/N <HoAj"xf tag
2=*=^)FNI 参见下面tag 元素
6XUuGxQV/ gtU1'p" Listener元素
IhonnLLW hq_~^/v\ 一个tag库可能定义一些类做为它的事件侦听类,这些类在TLD中被称为listener 元素,jsp服务器将会实例化这些侦听类,并且注册它们。Listener元素中有一个叫listener-class的子元素,这个元素的值必须是该侦听类的完整类名。
/lD?VE W|c.l{A5Q Tag元素
<+E%E4 2hOPzv&B 每个tag元素在tag库中都要指出它的名字、类名、脚本变量、tag的属性。其中脚本变量的值可以直接在TLD中定义或通过tag附加信息的类来取得。每个属性描述了这个属性是否可以省略,它的值是否可以通过<%= …%>这样的JSP语法来获得,以及属性的类型。
e1 a*'T$z +cg
{[f,J; 每一个tag在TLD中对应一个tag元素,下表是tag元素的子元素:
vE\lp8j+ CMVS W6 Tag元素的子元素
\ElX~$fS Dx+K+( 元素名称
3;/?q 描述
u}jrfKdE "n?<2
wso name
YB!!/ SX4 独一无二的元素名
Wc'Ehyi; :$H!@n*/R tag-class
'Ji+c Tag标签对应的tag处理类
>Q2). E =HE
m) tei-class
gg]~2f javax.servlet.jsp.tagext.TagExtraInfo的子类,用于表达脚本变量(可选)
X%4h(7;v <MZi<Z` body-content
b7$}JCn Tag标签body的类型
(K
#A EF;,Gjh5p display-name
B=zMYi 被可视化工具(诸如Jbuilder)用来显示的名称(可选)
~775soN iHz[Zw^.s small-icon
DP>mNE 被可视化工具(诸如Jbuilder)用来显示的小图标(可选)
h([0,:\ :C%47qv large-icon
nFXAF!,jj 被可视化工具(诸如Jbuilder)用来显示的大图标(可选)
YvTA+yL xH$%5@~ description
k>{-[X,/OV 此tag标签的描述
Y><(? @hT;Bo2G] variable
yiT{+;g^ 提供脚本变量的信息(同tei-class)(可选)
u2lmwE )73DT3-0$ attribute
ay[+2" Tag标签的属性名
(3fPt;U A?Sm-#n{ 以下章节介绍对于不同类型的tag,如何具体地实现它们。
\{>eOD_ ?UK:sF|(O %HEmi; 简单的tag
F*<Ws;j \X opU" >sV Bj(f tag处理类
r}@< K 2{};6{yz 简单的tag处理类必须实现Tag接口的doStartTag和doEndTag方法。当jsp引擎碰到tag标签的开头时,doStartTag被调用,因为简单的tag没有body,所以此方法将返回 SKIP_BODY。当jsp引擎碰到tag标签的结尾时,doEndTag被调用,如果余下的页面还要被计算,那它将返回EVAL_PAGE,否则将会返回SKIP_PAGE。
XI
g|G}i. w4}(Ab<Y 以下是例子:对于标签 <tt:simple /> ,它的tag处理类实现如下:
lt%-m@#/ w:R]!e_6\9 SRk7gfP*q public SimpleTag extends TagSupport
AzX(~Qc 6SAQDE {
Na;t#, v<fWc971 public int doStartTag() throws JspException
/O"0L/hc^ K!b>TICa: {
-9Xw]I#QR CU !.!cZ{ try{
<Am^z~[ m2MPWy5s pageContext.getOut().print(“Hello.”);
I)uASfT$ KqY>4tb }catch(Exception e){
XAlD
ww k`Y,KuBpM throw new JspTagException(“SimpleTag: “ + e.getMessage());
RA~_]Hk c=<v.J@K }
/K&