社区应用 最新帖子 精华区 社区服务 会员列表 统计排行 社区论坛任务 迷你宠物
  • 3895阅读
  • 1回复

一篇介绍JSP标签库很详细的文章,可以做为参考手册

级别: 终身会员
发帖
3743
铜板
8
人品值
493
贡献值
9
交易币
0
好评度
3746
信誉值
0
金币
0
所在楼道
标准的JSP 标记可以调用JavaBeans组件或者执行客户的请求,这大大降低了JSP开发的复杂度和维护量。JSP技术也允许你自定义taglib,其实换句话说,taglib可以看成是对JSP标记的一种扩展,正如xml是对html的一种扩展一样。taglib通常定义在tag标签库中,这种标签库存放着你自己定义的tag标签。简而言之,如果使用taglib,那么你可以设计自己的JSP标记! kZlRS^6  
K 7x,>  
一般来说,自定义tag标签主要用于操作隐藏对象、处理html提交表单、访问数据库或其它企业级的服务,诸如邮件和目录操作等等。自定义tag标签的使用者一般都是那些对java编程语言非常精通,而且对数据访问和企业级服务访问都非常熟悉的程序员,对于HTML设计者来说,使得他可以不去关注那些较复杂的商业逻辑,而将精力放在网页设计上。同时,它也将库开发者和库使用者进行合理分工,自定义tag标签将那些重复工作进行封装,从而大大提高了生产力,而且可以使得tag库可用于不同的项目中,完美地体现了软件复用的思想。 YS_9M Pi  
(5"BKu1t  
在这篇文章中,我们主要讨论: >:ZlYZ6sI  
us.+nnd  
· 什么是自定义tag标签? l7]$Wc[  
!6wbg  
· 怎么使用tag标签? VH{SE7  
'9,14e6   
o 声明要使用的tag库 Xcq 9*!%o  
!n`ogzOh  
o 找到与之对应的tag处理类 RfT#kh/5  
|\/0S  
o tag标签的类型 ==psPyLF@  
C/P,W>8  
· 自定义tag标签 RRD\V3C84  
6, Q{/  
o tag处理类 y>S.B/ d  
"9[K  
o tag库描述 zXB]Bf3TH  
ta+'*@V +G  
o tag标签示例 ; y#6Nx,:  
c;w~-7Q*|  
o 带属性的tag b8 J\Lm|J  
9?*BN\E5S  
o 带body的tag *JG?^G"l  
 ?4 `K8  
o 定义了脚本变量的tag 4/{Io &|  
lLHHuQpuj  
o 具有协作关系的tag ?ng?>!  
_%s_w)  
· 自定义tag标签 b`n+[UCPtn  
Rg&6J#h  
o 一个迭代tag的例子 !oU$(,#9  
Tk-PCra  
o 一个模板tag库 %E7+W{?*1  
tg~&kaz  
o tag处理类到底是怎样被调用的? *DX6m  
V/+r"le  
&Ko}Pv  
什么是自定义的tag? w\Q3h`.  
3V`K^X3  
一个自定义的tag标签是用户定义的一种JSP标记。当一个含有自定义的tag标签的JSP页面被jsp引擎编译成servlet时,tag标签被转化成了对一个称为tag处理类的对象进行的操作。于是当JSP页面被jsp引擎转化为servlet后,实际上tag标签被转化成为了对tag处理类的操作。 RlX;c!K  
%^"Tz,f  
自定义tag标签有很多特色,诸如: ]}9y>+>  
(FMYR8H*(  
· 可以在JSP页面中自定义tag标签的属性 w2~(/RgO  
BzA(yCu$:  
· 访问JSP页面中的所有对象 mf@YmKbp  
##u+[ !  
· 可以动态地修改页面输出 z$-/yT"M  
_\"P<+!  
· 彼此这间可以相互通信。你可以先创建一个JavaBeans组件,然后在一个tag中调用此JavaBeans组件,同时可以在另一个tag中调用它。 /[[zAq{OA  
j&fr4t3  
· tag允许相互嵌套,可以在一个JSP页面中完成一些复杂的交互。 _dsd{&  
5%?La`C9[  
ru(Xeojv#  
使用tag标签 `+B+RQl}[  
D{.%Dr?  
p`1d'n[  
本节主要描述怎样在JSP页面中使用tag标签,以及tag标签的不同类型。 ]Nt97eD)  
]-%ZN+  
要使用tag标签,JSP程序员必须做2件事: }#<Sq57n  
o^&; `XOd  
· 声明此tag标签的tag库 Eg-b5Z);  
8+gx?pb  
· 实现此tag标签 y%TR2CvT  
4g\a$7 r  
声明tag标签所在的tag库 1a4QWGpq  
92Ar0j]  
如果要使用tag标签,则应用JSP的taglib指示符来指定其tag库(注意:taglib要在在使用此tag标签之前声明) 7@EYF  
bq[Q  
<%@ taglib uri=”/WEB-INF/tutorial-template.tld” prefix=”tt” %> J~gfMp.  
N>ct`a)BD/  
uri属性定义了唯一的标签库描述(以下简称TLD),它可以是直接是tld文件名或一个独一无二的名字。prefix是用来区别其它TLD中和本TLD中有重名的tag的一种手段。 a4`@z:l  
8ZG'?A+{  
TLD必须以.tld作为扩展名,并且存放在当前应用的WEB-INF目录或其子目录下。你可以通过它的文件名直接引用它,也可以通过别的方式间接地引用它。 dN |w;|M  
{k?Y :  
以下taglib指示符直接引用一个TLD: -&,NM  
5\=9&{WjND  
<%@ taglib uri=”/WEB-INF/tutorial-template.tld” prefix=”tt” %> &1Cq+YpI  
1LPfn(  
以下的taglib指示符通过一个逻辑名称间接地引用一个TLD: XZ|"7as  
zo} SS[  
<%@ taglib uri=”/tutorial-template” prefix=”tt” %> & p_;&P_  
qZ\zsOnp  
如果是间接引用TLD的话,那你必须还要在web.xml中定义此逻辑名称与tld文件之间的映射,具体做法是在web.xml中加入一个名为taglib的元素: @/$mZ]|T  
_hh|/4(  
<taglib> K7f-g]Ibdn  
R@VO3zsW  
<taglib-uri>/tutorial-template</taglib-uri> n;Iey[7_E`  
d}l^yln  
<taglib-location> sM5 w~R>Y  
-S,dG|  
/WEB-INF/tutorial-template.tld hq,;H40%/  
TJ; v}HSo  
</taglib-location> )D[xY0Y~  
:>3&"T.  
</taglib> f2yv7t T   
oC7#6W:@w  
~!&[;EM<bm  
实现此tag标签 }SN'*w@E  
FwW%@Y  
jQ^Ib]"K  
为了实现tag标签,你有2种方法来存放tag处理类。一、让tag处理类以.class的方式存放于当前应用的WEB-INF/class子目录下,二、如果tag处理类是以JAR包的形式存在的话,那可以放在当前应用的WEB-INF/lib目录下,如果tag处理类要在多个应用中共享,那么它就应放在jsp服务器上的common/lib目录下,对于tomcat来说,就是tomcat/common/lib目录下。 q yJpm{  
23OV y^b  
^U`q1Pg5  
tag标签类型 ^_=0.:QaW  
zcZw}  
rSJ}qRXwU  
自定义的tag标签遵循XML语法。它有一个开始标记和一个结束标记,有的还有body(即文本节点): A.h?#%TLL  
D$mrnm4d  
<tt:tag> #TSM#Uqe  
^Q OvK>W<  
body {(q U n  
=f|>7m.p  
</tt:tag> Q` s(T  
+nZG!nP  
2 sOc]L:9  
一个不带body的tag标签如下: l|fd,  
(,TO|  
<tt:tag /> 3AL.UBj&}  
 WcJ{}V9  
GbE3 :;JI  
简单的tag标签 ]e3nnS1*.  
:Aa^afjJw  
一个没有body和属性的tag标签如下: :_H88/?RR  
.iYgRW=T  
<tt:simple /> 8xv\Zj+  
A^Zs?<C-  
' FK"-)s  
带属性的tag标签 8`~]9ej  
|S8pq4eKJ_  
d{m0uX56  
自定义标签可以有自己的属性。属性一般在开始标记中定义,语法为 attr=”value”。属性的作用相当于自定义标签的一个参数,它影响着tag处理类的行为。你可以在TLD中详细定义它。 <@=NDUI3*,  
a&[nVu+  
你可以用一个String常量给一个属性赋值,也可以通过表达式给它赋值,如<%= ...%>。以struts为例,它的logic:present标签就是用的String常量来给属性赋值: MDq@:t  
,@;<u'1\G  
<loglic:present parameter = “Clear”> o&O!Ur  
xVe!  
而另一个标签logic:iterate是用表达式来给属性赋值: Q4PXC$u  
?`XKaD! f  
<logci:iterate collection=”<%= bookDB.getBooks() %>” =3:ltI.'*I  
`otQ'e~+t  
id=”book” type=”database.BookDetails”> r=`>'3 } x  
<f6Oj`{f4  
3YRzBf:h  
带body的tag标签 8HOmWQS  
;1 fML,8  
一个自定义标签可以包含其它自定义标签、脚本变量、HTML标记或其它内容。 %xRS9A 4  
<b\urtoJ  
在下述例子中,此JSP页面使用了struts的logic:present标签,如果些标签定义了parameter=”Clear”的属性,则将清除购物车的内容,然后打印出一条信息: rij[ZrJ  
ZZ(@:F  
<logic:present parameter=”Clear”> 1}pR')YL[  
Q'VS]n  
<% cart.clear(); %> #o RUH8  
W#[!8d35$  
<font color=”#ff0000” size=”+2”><strong> nGVqVSxKT  
'w2;oO  
你选择了清除购物车! f- <6T  
Exr7vL  
</strong></font> 7 'S]  
`{Jb{L@f  
</logic:present> qAS^5|(b[  
E@Fen CF  
=+ p+_}C  
到底是用属性还是用body来传递信息? |@BN+o;`Om  
YSgF'qq\  
如上所述,我们既可以通过属性,也可以通过body来传递信息。但一般来说,比较简单的类型,如字符串或简单表达式最好采用属性来传递信息。 g8Z14'Ke  
J%)2,szn0  
d%3BJ+J  
定义脚本变量的tag标签 `cpcO  
Pcs@`&}7r  
所谓脚本变量,是指JSP中可以调用的变量或对象。它可由tag标签产生。以下示例阐述了一个tag标签定义了一个名为tx的由JNDI所定义的事务处理对象。脚本变量可以是ejb对象、事务、数据库连接等等: ja/[PHq"  
+b+sQ<w?.  
<tt:lookup id=”tx” type=”UserTransaction” name=”java:comp/UserTransaction” /> ^&iV%vQ[  
3U)8P6Fz  
<% tx.begin(); %> <Xx\F56zp  
%5%Wo(W'  
... 9f,:j  
zwtsw[.  
.TC `\mV  
具有协作关系的tag标签 Ao T7sy7  
#=(op?]  
自定义tag标签之间可以通过共享对象来实现协作。在下述例子中,标签tag1创建了一个名为obj1的对象,在标签tag2仍可以重复使用obj。 Z<W6Avr  
h>:RCpC  
<tt:tag1 attr1=”obj1” value1=”value” /> wVDB?gy%#  
7h1gU  
<tt:tag2 attr1=”obj1” /> )_cv}.xe  
pm\X*t}L  
在以下这个例子当中,如果外层的tag标签创建了一个对象,那么其内层的所有tag标签都可以使用这个对象。由于这样产生的对象没有一个指定的名字,那么就可以将少重名的冲突。这个例子阐述了一系列协作的嵌套对象。 Wg^cj:&`u  
2(Uz9!<V  
<tt:outerTag> 7-I>5 3@  
 c^s>  
<tt:innerTag /> 5#.\pR{Gd  
RFY!o<   
</tt:outerTag> p^P y,  
9 w1ONw8v  
KVxb"|[  
Tag处理类 D 'cY7P  
<C;> $kX  
{yMA7W7]  
Tag处理类必须实现Tag接口或BodyTag接口,不过现在一般都流行从TagSupport或BodyTagSupport类中继承,这些类或接口都可以在javax.servlet.jsp.tagext包中找到。 o '/C$E4W  
x9l0UD*+g  
当JSP引擎看到自己的JSP页面中包含有tag标签时,它会调用doStartTag方法来处理tag标签的开头,调用doEndTag方法来处理tag标签的结束。 bpa O`[*  
?VQLY=?  
下表说明不同类型的tag所需要不同的处理过程: :xS&Y\ry  
8G&+  
Tag处理类的方法 f\|?_k]  
Fx5d@WNa>  
Tag标签类型 D1~x  
所调用的方法 F*t_lN5{  
w/5^R  
基本标签 wI@I(r~ g  
doStartTag, doEndTag, release f&|A[i>g  
T% 13 '  
带属性的标签 e~.?:7t  
doStartTag, doEndTag, set/getAttribute1...N, release 6h6?BQSE  
cO+Xzd;838  
带内容的标签 9<h]OXv  
doStartTag, doEndTag, release 'z}M[h K]  
l@r wf$-  
带内容的标签,且内容重复循环 !L. K)9I  
doStartTag, doAfterBody, doEndTag, release 4:&qT Y)H  
q%k(M[  
带内容的标签,且内容与JSP交互 `k]!6osZo  
doStartTag, doEndTag, release, doInitBody, doAfterBody, release L'XdX\5  
uO@3vY',n  
一个tag处理类可以通过javax.servlet.jsp.PageContext来与JSP交互,通过javax.servlet.jsp.PageContext类,tag处理类可以访问JSP中的request、session和application对像。 :q64K?X  
|"YA<e %  
如果tag标签是互相嵌套的,那内层的tag处理类可以通过它的parent属性来访问上层的tag处理类。 MiH}VfI  
FaE#\Q  
一般情况都将所有的tag处理类打成了JAR的包,以便于发布。 *UBP]w  
BBR" HMa4  
)R8%'X;U  
Tag库描述(简称TLD) KBw9(  
WAB0e~e:|Q  
VkDS&g~Ws  
Tag库是用xml语言描述的,TLD包括了tag库中所有tag标签的描述,它一般用来被jsp服务器用来校验tag的语法正确性,或者被jsp开发者用来开发新的标签。 .6ngo0<g   
>$naTSJq  
TLD的文件扩展名必须为.tld,而且必须放在当前WEB应用的WEB-INF目录或其子目录中。 mPfUJ#rS  
!Wixs]od   
一个TLD的内容的开头必须遵守标准的XML开头,用于描述DTD和xml的版本,例如: /pN'K5@  
WVWS7N\  
<?xml version="1.0" encoding="ISO-8859-1" ?> 7AO3-; l]  
Bhu@ 2KdA  
<!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN" "http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd"> HS\3)Ooj>  
TLD必须以<taglib>来作为它的根元素,<taglib>的子元素如下表: =,4 '"  
p2x1xv  
,9Z2cgXwJ  
<taglib>的子元素 z^rhgs?4  
~mK-8U4>K,  
Element k%]DT.cE  
Description FE+7X=y  
ZUXse1,  
tlib-version m0|Ae@g~3  
Tag库的版本 4@v1jJj  
6Lg!L odu  
jsp-version df4sOqU  
Tag库所需要的jsp的版本 \5Vp6^  
L1 9 MP  
short-name Nmp>UE,7[  
助记符,tag的一个别名(可选) +ze}0lrEL  
YX ;n6~y  
uri [,<\RviI  
用于确定一个唯一的tag库 \P l,' 1%  
jby~AJf %  
display-name W$" Y%^L  
被可视化工具(诸如Jbuilder)用来显示的名称(可选) R:w %2Y  
]{!!7Zz  
small-icon psIkG0 &  
被可视化工具(诸如Jbuilder)用来显示的小图标(可选) 2{t)DUs  
(Sth:{;  
large-icon T?X^0UdJj  
被可视化工具(诸如Jbuilder)用来显示的大图标(可选) ':l"mkd+`  
7qP4B9S  
description Fn:.Y8%-  
对tag库的描述(可选) rDVgk6  
$gVLk.  
listener !2}rtDE  
参见下面listener元素 uR#'lb`3  
<S]KaDu^  
tag cz2,",+~  
参见下面tag 元素 - <J q  
yp=|7  
Listener元素 }{ n\tzR  
0iYP  
一个tag库可能定义一些类做为它的事件侦听类,这些类在TLD中被称为listener 元素,jsp服务器将会实例化这些侦听类,并且注册它们。Listener元素中有一个叫listener-class的子元素,这个元素的值必须是该侦听类的完整类名。 u3]Uxy  
Z^s+vi  
Tag元素 6nh]*/  
>b>M Km>q  
每个tag元素在tag库中都要指出它的名字、类名、脚本变量、tag的属性。其中脚本变量的值可以直接在TLD中定义或通过tag附加信息的类来取得。每个属性描述了这个属性是否可以省略,它的值是否可以通过<%= …%>这样的JSP语法来获得,以及属性的类型。 7}Bj|]b)~  
n`%2Mj c  
每一个tag在TLD中对应一个tag元素,下表是tag元素的子元素: 6|,e%  
i{8]'fM  
Tag元素的子元素 "H>.':c"+3  
\V= &&(n#  
元素名称 cVp[ Z#B  
描述 A2y6UzLYD  
&K>cW$h=a  
name  kI%peb?  
独一无二的元素名 h40'@u^W  
~n WsP}`n  
tag-class M;*$gV<x  
Tag标签对应的tag处理类 Va7c#P?  
R :"+ #Sq  
tei-class XYjcJ  
javax.servlet.jsp.tagext.TagExtraInfo的子类,用于表达脚本变量(可选) dIk' pA^d  
/}Yqf`CZy  
body-content s`"OM^[-  
Tag标签body的类型 |8H_-n  
Bz+.Qa+  
display-name q +R*Hi  
被可视化工具(诸如Jbuilder)用来显示的名称(可选) 'w(y J  
-/>9c-F  
small-icon FEF"\O|Q  
被可视化工具(诸如Jbuilder)用来显示的小图标(可选) 9JPEj-3`g  
qe$33f*  
large-icon 2Z)4(,  
被可视化工具(诸如Jbuilder)用来显示的大图标(可选) ["a"x>X&  
EUy(T1Cl&&  
description rfs(#  
此tag标签的描述 Sh&iQ_vq  
mG2VZ>  
variable GRT] aw  
提供脚本变量的信息(同tei-class)(可选) O$qtq(Q%  
:= <0=JE#  
attribute %X>FVlPm  
Tag标签的属性名 _A{+H^,  
XE1$K_m  
以下章节介绍对于不同类型的tag,如何具体地实现它们。 ~U$":~H[  
R)oB!$k  
R4{-Qv#8 q  
简单的tag {,?ss$L  
sWHyL(C@  
zrRFn `B  
tag处理类 ZyI$M3{J  
I{ ryD -!  
简单的tag处理类必须实现Tag接口的doStartTag和doEndTag方法。当jsp引擎碰到tag标签的开头时,doStartTag被调用,因为简单的tag没有body,所以此方法将返回 SKIP_BODY。当jsp引擎碰到tag标签的结尾时,doEndTag被调用,如果余下的页面还要被计算,那它将返回EVAL_PAGE,否则将会返回SKIP_PAGE。 &nY;=Hv`WY  
f+x ;:  
以下是例子:对于标签 <tt:simple /> ,它的tag处理类实现如下: Fi2xr<7"  
0{+.H_f`  
,%[4j9#!_  
public SimpleTag extends TagSupport '\"G{jU@  
XF*.Jg]  
{ rrqQCn9  
}D*yr3b  
public int doStartTag() throws JspException s5D:  
=']3(6*  
{ bgF^(T35  
O_;Dk W  
try{ kn"q:aD  
!eI2 r   
pageContext.getOut().print(“Hello.”); $f$|6jM  
6N~q`;p0  
}catch(Exception e){ ZmEEj-*7s  
Lrd[O v  
throw new JspTagException(“SimpleTag: “ + e.getMessage()); a.2L*>p  
R>/ NE!q  
} vLT0ETHg6  
n,$z>  
return SKIP_BODY; 4J0Rv od_  
xSMt*]=9  
} LP];x3  
o-C#|t3hH  
public int doEndTag() X4|4QgY  
;# uZhd  
{ @-&MA)SN  
Yg9joNBh  
return EVAL_PAGE; n.}E5 %qK  
@Fp_^5  
} 6 Orum/|h  
kE9esC 3  
} Qv,"($n\  
#|&Sc_#4)  
nw>8GivO  
注意:如果tag标签没有内容的话,那必须定义body-content元素为空,例如 ojvj}ln  
eIRLNxt+v  
<body-content>empty</body-content> VXIP0p@  
9l,a^@Y:  
p$Ox'A4  
带属性的tag标签
评价一下你浏览此帖子的感受

精彩

感动

搞笑

开心

愤怒

无聊

灌水
级别: 终身会员
发帖
3743
铜板
8
人品值
493
贡献值
9
交易币
0
好评度
3746
信誉值
0
金币
0
所在楼道
只看该作者 1 发表于: 2007-01-01
tag处理类 ,AxdCT  
]]Cb$$Td  
对于tag标签的每个属性,你必须依照JavaBeans规范来定义其属性,以及get和set方法。以struts的logic:present 标签为例, B!vmQR*1  
#Q2Y&2`yGT  
<logic:present parameter=”Clear”> $STaQ28C  
:<}=e@/~|  
与此相应,此tag处理类应有如下方法和定义: 5$V_Hj  
anXc|  
/YZr~|65  
protected String parameter = null; 0q&<bV:D  
.zi_[  
public String getParameter() {NHdyc$  
0aG ni|  
{ Ney/[3 A  
q@[Qj Gj@  
return this.parameter; 8_{X1bj  
~`aa5;Ab_  
} eEuvl`&  
BO;6 u^[  
public void setParameter(String parameter) rJGf .qJJ  
Wk)OkIFR  
{ D)L+7N0D~  
~_/(t'9  
this.parameter = parameter; vEJWFoeEFm  
E*]bgD7V  
} gt@m?w(  
59h)-^!  
G3Z)Z) N  
注意:如果你的属性名为id,而且你的tag处理类是从TagSupport类继承的,那你就不需要定义它的属性和set和get方法,因为他们早已在TagSupport被定义过了。 }H^+A77v  
E=nIRG|g  
Attribute元素 <g$~1fa  
17%Mw@+  
对于tag标签的每个属性,你必须定义它是否必须的,它的值是否可以用诸如<%= …%>的表达式来获得,以及它的类型(可选),如果不指定它的类型,那就默认为是java.lang.String类型。如果rtexprvalue元素被定义为true或yes,那么在type元素中就定义了attribute的返回类型。 <0Xf9a8>  
|?,A]|j  
<attribute> f4fvrL  
'Gj3:-xqL  
<name>attr1</name> :tV*7S=)  
jIJ~QpNE  
<required>true|false|yes|no</required> 3LOdjT J  
fdFo#P  
<rtexprvalue>true|false|yes|no</rtexprvalue> hgPa6Kd  
0Tx6zO  
<type>attribute的返回类型(只用当rtexprvalue为真是才有效)</type> Ayxkv)%:@)  
d3\qKL!~  
</attribute> T&6l$1J  
eA2@Nkw~)  
如果tag的某个属性不是必须的,那tag处理类会自动提供一个缺省值。 k\5c|Wq|g  
v[1aW v:  
例如,在logic:present这个tag标签中定义了一个属性叫parameter,但它不是必须的,而且它可以被诸如<%= …%>的表达式来赋值。 H#,W5EJzM  
'jWr<]3  
<tag> M:Pc,  
TM%| '^)  
<name>present</name> Fs9!S a7v  
01t1Z}!y  
<tag-class>org.apache.struts.taglib.logic.PresentTag</tag-class> |d{PA.@33  
&."iFe  
<body-content>JSP</body-content> ,Vc6Gwm  
"L IF.)  
y%"{I7!A  
/:m-> T  
<attribute> (KjoSN( K  
9+Np4i@  
<name>parameter</name> "H'B*vc-  
3dg1DR;  
<required>false</required> ;17E(tl  
L>jY.d2w=K  
<rtexprvalue>true</rtexprvalue>  dm\F  
,0 M_ Bk"  
</attribute> WlOmJtt4)  
03$mYS_?  
^UP`%egR  
LBw1g<&  
</tag> 0"jY.*_EW  
NVkV7y X]  
~[t[y~Hup  
属性元素的校验 3#LlDC_WC  
Ouk ^O}W6  
Kg{+T`  
有关于tag标签的有效值可以从tag库的说明文档中获得,当JSP页面被编译时,jsp引擎会强制性地参照TLD中定义的规则进行检查。 (p"%O  
)8a~L8oN  
还有一个方法也可以进行属性元素的校验,就是先继承类TagExtraInfo,然后调用它的isValid方法。这个类同时也起到提供tag中定义的脚本变量信息的作用。 !z\h| wU+  
">\?&0  
IsValid方法通过TagData对象来传递属性信息,它包括着tag的所有的属性名-值的信息。由于校验发生在运行时刻,因此这个属性的值将被赋值为TagData.REQUEST_TIME_VALUE。 T^zXt?  
8?xE6  
例如tag标签<tt:twa attr1=”value1” />在TLD中定义如下: g@d*\ P)  
9SX +  
<attribute> 0_jf/an,%  
U7?;UCmX  
<name>attr1</name> Akq2 d;  
fW?vdYF  
<required>true</required> &h}#HS>l  
]vAz  
<rtexprvalue>true</rtexprvalue> * ` JYC  
l[dK[4  
</attribute> +Q/R{#O  
D, k6$`  
))qy;Q,  
这个定义说明了attr1能在运行期间被赋值。 lt/1f{v[:  
6,pnw  
以下的isValid方法检查attr1属性的值是否属于Boolean类型。注意由于attr1能在运行刻被赋值,那么isValid方法必须检查tag用户是否对此tag进行了运行时刻赋值。 ,V7nzhA2  
8=!D$t\3  
?al'F  q  
Public class TwaTEI extends TagExtraInfo N:^n('U&j  
!hm]fh_j  
{ rjK%t|aV^  
pJ>P[  
public boolean isValid(Tagdata data) +5)nk}  
^vZSUfS  
{ <;lkUU(WT2  
)7hqJa-V  
Object o = data.getAttribute(“attr1”); Kf3"Wf^q   
}H53~@WP>  
If(o != null && o != TagData.REQUEST_TIME_VALUE) Fw_#N6Q  
!M(xG%M-V  
{ jV i) Efy  
Zj(AJ*r  
if( ( (String)o).toLowerCase().equals(“true”) || b 1c y$I  
5%"V[lDx@  
((String)o).toLowerCase().equals(“false”) ) H8=N@l  
$z6_@`[  
return true; `>o{P/HN  
a .#)G[*  
else Zpt\p7WQ  
@NR>{Eg  
return false; hW<%R]^|  
=nHUs1rKn  
} gt) I(  
}{Pp]*I<A  
else 9X6h  
1C+13LE$U  
return true; {p2!|A&a  
cVv=*81\  
} M/f<A$xx_  
AYBns]!  
} =jN.1}  
`7E;VL^Y1  
, >a&"V^k  
带body的tag jVe1b1rt~3  
B`)BZ,#p  
w{8xpAqm  
tag处理类 l:~/<`o  
Oi.C(@^(  
,L2ZinU:  
如果tag标签含有内容,那处理方式会略微有些不同,而且还要视tag处理类是否要与body交互的情况而定。如果要与body交互,那我们认为tag处理类要可能要对body进行操作。 BKCiIfkZ  
,_ H:J.ik  
Tag处理类不与body交互 v4TQX<0s  
^LnTOdAE  
如果tag处理类不与body交互,tag处理类应该实现Tag接口或从TagSupport中派生,如果body要被计算,那么doStartTag方法应返回 EVAL_BODY_INCLUDE,否则应返回SKIP_BODY。  tU5zF.%  
gx/,)> E.  
如果tag处理类要对body反复运算,则它应该实现IterationTag或从TagSupport中派生。如果tag处理类认为body还未计算完的话,那它的doStartTag方法和doAfterBody方法必须返回EVAL_BODY_AGAIN。 Y1\}5k{>  
e(&v"}Ef`  
Tag处理类与body交互 eS^7A}*wd-  
9.M4o[  
如果tag处理类与body交互,那tag处理类应实现BodyTag接口或从BodyTagSupport中派生。这种tag处理类一般要实现doInitBody和doAfterBody方法。 HVCe;eI  
C3f' {}  
Body允许一些方法来读写它的内容。Tag处理类可以调用body内容的getString或getReader方法来从body中提取信息,也可用 writeOut(out) 方法来将body的内容写入到out对象中。其中out对象通过tag处理类的getPreviousOut方法来获得。 DCO\c9  
!PlEO 2at  
如果body的内容需要被计算,那么doStartTag方法必须返回EVAL_BODY_BUFFERED,否则,它将返回 SKIP_BODY。 p<FzJ   
(S5R!lpO  
doInitBody 方法 D/gw .XYL  
 Mx?d  
此方法在body内容已经设好,但未被计算之前被调用。你可以根据不同的body内容来制定初始化策略。 n38p!oS  
3ZPWze6  
doAfterBody方法 7aRi5  
u~:y\/Y6  
此方法在body内容已被计算后进行调用。 wW P}C D  
1-uxC^u?|#  
和doStartTag方法一样,doAfterBody方法返回一个指示符指示是否要继续计算body,如果要继续计算,则doAfterBody应返回EVAL_BODY_BUFFERED,否则,它应返回SKIP_BODY。 ,-LwtePJ0  
#Vt%@* i  
release 方法 bH9kj/q\b  
| j`@eF/"  
tag处理类调用此方法将它的状态重置为初始状态,并释放所有的私有资源。 1=c\Rr9]  
x+:UN'"r  
OZF rtc+  
以下的例子读取body的内容(其中含有一条sql语句),然后将它传递给一个对象,让它进行查询。由于此处body不须重新计算,所以doAfterBody会返回SKIP_BODY。 pj{`'; :g  
IMFDM."s  
U$.@]F4&  
Public class QueryTag extends BodyTagSupport d L 1tl  
8W(*~}ydYY  
{ D/xbF`  
_Ey9G  
public int doAfterBody() throws JspTagException $9#H04.x  
~ 'cmSiz-  
{ \ZFGw&yN  
(Z q/  
BodyContent bc = getBodyContent(); 6I4\q.^qw  
.GXBc  
//将body的内容以字符串的格式提取出来 GW@;}m(  
zT.7  
String query = bc.getString(); 4r#= *  
3{64 @s  
//清除body 6r0krbN  
ZohCP  
bc.clearBody(); Oc#syfO  
G@\1E+Ip  
try{ IB"w&sBy  
(O?.)jEW(.  
Statement stmt = connection.createStatement(); sN*N&XG  
z"L/G  
Result result = stmt.executeQuery(query); DB|Y  
m+R[#GE8#  
}catch(SQLException e){ |Nn)m  
K~{$oD7!  
throw new JspTagException(“queryTag: “ + e.getMessage() ); X6X $Pve  
,/%=sux  
return SKIP_BODY; Su7?;Oh/yI  
m|n%$$S&  
} }Yzco52  
HY*Kb+[  
} H3 ^},.  
mt{nm[D!Xp  
4<Utmr  
body-content元素 =JEv,ZGT3  
^R7lom.  
QL&ZjSN  
由于tag可能会有body,你必须用body-content元素来指定body内容的类型: 36Zf^cFJ  
uM IIYS  
<body-content>JSP|tagdependent</body-content> JN-y)L/>  
q460iL7yF}  
{yHCXFWlS  
如果body的内容是定制的或内部的tag、脚本元素、或HTML广本,则归类为JSP类型。其他的类型,比如上面代码所述的?D?D将sql statement类传给 query tag的这种类型应该标为tagdependent。 w !-gJmX>  
JV^=v@Z3  
注意:实际上body-content的值并不影响tag处理类对body内容的处理,它仅仅是被tag编辑工具用来描述此body的内容。 {OkV%Q<  
%~H-)_d20  
yy^q2P  
用tags定义脚本变量 qs6]-  
6 l|DU7i  
x;P_1J%Q  
tag处理类 mZS >O_E  
XH4  
{ M4gF8(M  
tag处理类负责创建或设置页面中定义的脚本变量,用pageContext.setAttribute(name,value,scope)或pageContext.setAttribute(name,value)方法来实现。一般来说,tag处理类通过脚本变量的名称来获取它,脚本变量的名称一般可用get方法来获得。 )Xz,j9GzJS  
QC OM_$y  
如果脚本变量的值依赖于tag处理类中的上下文中某一对象,那它可用pageContext.getAttribute(name,scope)方法来找到那个对象。一般的处理过程是tag处理类先找到脚本变量,再对其进行处理,然后用pageContext.setAttribute(name,object)的方法来设置它的新值。 KK/tu+"  
kzQ+j8.,U  
对象的生存周期(scope)如下表: +s,=lL  
|}s*E_/[  
对象的生存周期表 NqazpB*  
u^ +7hkk  
名字 D09Sg%w  
可访问范围 r;.yz I  
生存周期 T= y}y  
vAF "n  
page }I+E\ <  
当前页面 n`B:;2X,  
一直有效,除非页面向客户提交响应或重定向到一个新页面 17%,7P9pg  
^B.5GK)!  
request GbY7_N  
当前页面或当前页面重定向到的页面 Y1W1=Uc uk  
一直有效,除非页面向客户提交响应 sQHv%]s 0  
q.^;!f1  
session ^+>laOzC`8  
当前页面或在同一浏览器窗口中的页面 i4Q@K,$  
一直有效,除非关闭当前浏览器、超时、网络故障 KEo ,m  
E1aHKjLQ  
application r4b 6 c  
整个web应用程序的所有请求 oQ/E}Zk@  
一直有效,除非发生网络故障、服务器故障 (&Kk7<#`  
T?CdZc.  
提供关于脚本变量的信息 4<w.8rR:A  
{ =9,n\85#  
以下示例定义了一个名为“book”的脚本变量,用来访问程序中关于书的信息: ~|D Ut   
A7Cm5>Y_S  
<bean:define id=”book” name=”bookDB” property=”bookDetails” type=”database.BookDetails” /> >UTBO|95y  
Wq D4YGN  
<font color=”red” size=”+2” > HTv2#  
"+R+6<"  
<%= messages.getString(“CartRemoved”) %> 1APe=tJ  
$D~0~gn~  
<strong><jsp:getProperty name=”book” property=”title” /></strong> 2. NN8PPD"  
[x=s(:qy  
</font> hE'-is@7  
i(rL|d+'  
当包含此tag的JSP页面被编译时,jsp引擎会自动生成关于此book的同步的代码(同步可以避免几个客户同时访问此book时造成的冲突),要生成同步代码,jsp引擎需要知道此脚本变量的如下信息: a_^\=&?'  
q"CVcLi9  
· 脚本变量名称 htO +z7  
.ljnDL/  
· 脚本变量所属的类 L:$ ,v^2  
8rAg \H3E  
· 此脚本变量是否引用了一个新的或已存在的对象 WH#1 zv  
> ym,{EHK  
· 此脚本变量的有效性 P[G)sA_"  
kf\PioD8  
有两种办法可以向jsp引擎提供关于脚本变量的信息:在TLD中定义variable子元素,或用tei-class子元素定义一个额外tag信息类。用variable最简单,但可能降低了一些灵活性。 q<x/Hat)  
g>E LGG |Q  
Variable元素 k{R>  
60^`JVGWH  
Variable元素有如下子元素: p;`>e>$  
j1Y~_  
· name-given ?D?D 给出的名字,是一个常量 4B8 oO  
XFVE>/H  
· name-from-attribute?D?D 属性名,在编译时给出的属性名 fh&nu"&  
v|)4ocFK  
name-given或name-from-attribute两者必须选一,但以下子元素是可选的: i<C*j4qQ  
UP$.+<vm  
· variable-class?D?D变量的类型,缺省为java.lang.String。 w8")w*9Lmg  
9d0@wq.  
· declare?D?D此脚本变量是否引用了一个新对象,缺省为True。 =g7x' kN  
nSDMOyj+  
· scope?D?D脚本变量的范围,缺省为NESTED。下表描述了scope的几种类型: p#ZCvPE;uH  
CCs%%U/=  
脚本变量的有效范围 $8)+XmsCr  
~TF:.8  
^2:p|:Bz!l  
有效性 Y Vt% 0  
方法 ]#i igPZ7  
@o].He@L<j  
NESTED B-RjMxX4>  
在tag标签的开始和结束之间 ].avItg  
如果tag处理类实现BodyTag接口,则在doInitBody和doAfterBody中调用,否则在doStartTag中调用 <)C#_w)-  
np|Sy;:  
AT_BEGIN M><yGaaX/  
从tag标签的开始一直到页面结束 JMCKcZ%N  
如果tag处理类实现BodyTag接口,则在doInitBody、doAfterBody和doEndTag中调用,否则在doStartTag和doEndTag中调用 g.k"]lP  
.r=4pQ@#  
AT_END ?> 9/#Nv  
从tag标签的结束一直到页面结束 rET\n(AJ  
在doEndTag中调用 x;O[c3I  
M5 LfRBO  
以struts为例,它的bean:define标签的实现遵循JSP1.1规范,此规范要求使用额外tag信息类来定义脚本变量。Variable元素是JSP1.2规范中加入的。以bean:define标签为例,你可以定义如下variable元素: ~gJwW+  
LRxZcxmy  
<tag> i]c!~`  
h:))@@7MJ  
<variable> ,hDW Ps2S  
4Co6(  
<name-from-attribute>id</name-from-attribute> IV~>I-rd  
+zqn<<9  
<variable-class>database.BookDetails</variable-class> 7uqzm  
"`/h#np  
<declare>true</declare> +uF>2b6'  
,C\i^>=  
<scope>AT_BEGIN</scope> (!u~CZ;  
^cC,.Fdw  
</variable> ^ 'MT0j  
93>jr<A  
</tag> zEX  
LtO!umM  
额外tag信息类 +yG~T  
tn\yI!a  
如果要定义一个额外tag信息类,你要继承javax.servlet.jsp.TagExtraInfo类。一个TagExtraInfo类必须实现getVariableInfo方法,此方法返回一个叫VariableInfo的数组类,它包括如下信息: /obfw^  
a@K%06A;'  
· 变量名 E:_ZA  
zy?|ODM  
· 变量所属类名 3@_xBz,I.  
0(}t8lc  
· 此变量是否引用了一个新对象 88O8wJN  
]"As1"  
· 此变量的有效范围 r.=K~A  
R{`(c/%8  
jsp引擎将一个名为data的参数传给getVariableInfo方法,data中包括tag标签中的所有“属性名?D?D属性值”对。它可以用来向VariableInfo对象提供脚本变量的名字和类名。 4/~E4"8  
gT{Q#C2Baw  
以struts为例,它在bean:define标签中定义了一个名为DefineTei的额外tag信息类,用来向脚本变量提供信息。由于脚本变量的名称(book)和类名(database.BookDetails)是通过tag标签的属性来传递的,它们一般定义在VariableInfo的构建代码中,并且可用data.getAttributeString方法来得到这些信息。如果要允许book脚本变量能在从tag开始直到整个JSP页面结束的范围内都可用的话,那它的范围应设为AT_BEGIN。如下所示: x M/+L:_<  
>{n,L6_ t  
IxN9&xa  
public class DefineTei extends TagExtraInfo v |,1[i{  
wYXQlxdy  
{ :wyno#8`-  
Vi$~-6n&  
public VariableInfo[] getVariableInfo(TagData data) "m$##X\  
IZ-1c1   
{ J9nX"Sb  
!<8W {LT  
String type = data.getAttributeString(“type”); ' ,wFTV&  
yNJ B oar  
If( type == null) gnf8 l?M  
oAVnK[EMq`  
type = “java.lang.Object”; wc@X.Q[  
e`_LEv  
return new VariableInfo[] { ;W )Y OT  
ij`w} V  
new VariableInfo(data.getAttributeString(“id”), MTh<|$   
9Q^r O26+  
type, K=Z|/Kkh  
)gUR@V>e2  
true, \fLMr\LL&  
vkV0On  
VariableInfo.AT_BEGIN) a 7 V-C  
2DDtu[}  
}; 'W^YM@  
cxC6n%!;y  
}  @tnz]^V  
K:[F%e  
} epe)a  
;%9|k U  
9!\B6=r y4  
注意:关于额外tag信息类的类名必须要在TLD中的tag标签下的tei-class子元素中定义。因此,DefineTei的tei-class中的定义看起来如下: DH!~ BB;  
<tei-class> OX7M8cmc+  
org.apache.struts.taglib.bean.DefineTagTei Yx%Hs5}8  
</tei-class> a$OE0zn`  
X=&ET)8-Y  
`UyG_;  
具有协作关系的tag '3tCH)s  
FIhk@TKa  
/& {A!.;  
tag通过共享对象来进行协作,JSP技术支持2种方式的对象共享。 1<@W6@]  
2 c{34:  
第一种方法是使用pageContext对象进行对象的共享(可支持JSP页面和tag处理类之间的共享),如果在一个tag处理类中要调用由另一个tag处理类创建的对象,可调用pageContext.getAttribute(name, scope)方法。 9ULQrq$?  
S!CC }3zw  
第二各方式的共享是对于tag之间有嵌套关系而言的,外层的tag所创建的对象对于内层的tag来说是可以共用的。这种形式的共享的好处是减少了可能存在的重名冲突。 CAWNDl4  
BoWg0*5xb  
要访问一个嵌套tag创建的对象,tag处理类必须先找到此嵌套tag对象,可用TagSupport的静态方法 TagSupport.findAncestorWithClass(from, class)或TagSupport.getParent方法。前者在当不确定此tag是否为嵌套tag对象时使用。一旦它的父类被找到,它就能访问其所有动态或静态创建的对象。静态创建的对象是父类的成员,而动态创建的对象可能是父类的私有对象。诸如此类的对象可以用tag处理类的setValue方法来保存,用getValue方法来获得。 (k.[GfCbD  
1N-\j0au  
下例阐述了以上两种共享对象的方法。在这个例子当中,一个查询tag检查一个名为connection的属性名是否在doStartTag中被设置。如果connection属性被设置,tag处理类从pageContext中得到这个connection对象。否则,此tag处理类先找到它的父tag处理类,然后从它的父tag处理类中找到connection对象。 Y\k#*\'Y~  
z'n:@E  
b94DJzL1z  
public class QueryTag extends BodyTagSupport n0 {i&[I~+  
9wwqcx)3(  
{ '[:D$q;  
~rKrpb]ow  
private String connectionId; I;|B.j  
sY Qk  
public int doStartTag() throws JspException _S1>j7RQo  
j{A y\n(  
{ "Ac-tzhE  
DV-d(@`K  
String cid = getConnection(); %s|Ely)  
X`>i& I]  
if(cid != null) E6ElNgL  
hx%v+/  
{ HyZqUb Ha  
ZhaP2pC%4  
//存在一个connection id,使用它。 v>)"HL"XG  
*)T^Ch D,  
connection = (Connection) pageContext.getAttribute(cid); #OD/$f_  
,m:.-iy?  
} & l&:`nsJ  
3yF,ak {Sl  
else i%]EEVmN  
,T$U'&;  
{ +gtbcF@rx  
'Aq{UGN  
ConnectionTag ancestorTag = (ConnectionTag)findAncestorWithClass(this, 06Sceq  
v%z=ysA  
ConnectionTag.class); NP3y+s  
[EXs  
if(ancestorTag == null) [D4SW#  
"$^ ~!1~  
{ WlC:l  
k"iOB-@B+  
throw new JspTagException(“一个没有connection属性的查询标签必须被一个connection标记嵌套。”); ?mxMk6w  
w-MCZwCr)  
} X51:  
N +_t-5  
connection = ancestorTag.getConnection(); 7*A],:-q  
>W+%8e  
} !ons]^km  
MaQqs=  
} :>f )g  
@,7GaK\  
} Ai?*s%8v  
,Uqs1#r  
joAv{Tc  
f+)L#>Gl?  
此查询标签在JSP页面中的调用形式可以从以下2种定义中任选一种: ZG8DIV\D7  
e2oa($9  
P{>!5|k  
<tt:connection id=”con01” ...> ... </tt:connection> )' cMYC  
yjJ5>cg  
<tt:query id=”balances” connection=”con01” > @:vwb\azVD  
`kXs;T6&  
SELECT account, balance FROM acct_table y/7\?qfTk  
xdt- ;w|  
where customer_num = <%= request.getCustno() %> Q\7h`d%)  
Ie#Bkw'*  
</tt:query> vr6w^&[c^  
A]oV"`f  
p]+Pkxz]'  
>@_^fw)  
pO3SUOP  
<tt:connection ...> Kn;"R:  
I-(zaqp@  
<x:query id=”balances”> SZ'R59Ee<  
flbd0NB  
SELECT account, balance FROM acct_table ;$wVu|&  
!?h;wR  
where customer_num = <%= request.getCustno() %> >SHhAEF  
^Y \"}D  
</x:query> SO'vp z{  
N<VJ(20y  
</tt:connection> y??XIsF  
x g  
vXZOy%$o  
与此同时,在TLD中必须指定connection属性为可选的,定义如下: '_FsvHQ  
f46t9dxp$  
<tag> PKiy5D*8p  
=-n}[Y}A  
... nmKp[-5  
[hv~o~q  
<attribute> eru.m+\  
r[iflBP  
<name>connection</name> ;[OH(!  
i<Zc"v;  
<required>false</required> VjZ|$k  
`b7t4d*  
</attribute> }WXi$(@v  
S_UIO.K  
</tag>
描述
快速回复

您目前还是游客,请 登录注册
温馨提示:欢迎交流讨论,请勿纯表情、纯引用!
认证码:
验证问题:
3+5=?,请输入中文答案:八 正确答案:八