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

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

级别: 终身会员
发帖
3743
铜板
8
人品值
493
贡献值
9
交易币
0
好评度
3746
信誉值
0
金币
0
所在楼道
标准的JSP 标记可以调用JavaBeans组件或者执行客户的请求,这大大降低了JSP开发的复杂度和维护量。JSP技术也允许你自定义taglib,其实换句话说,taglib可以看成是对JSP标记的一种扩展,正如xml是对html的一种扩展一样。taglib通常定义在tag标签库中,这种标签库存放着你自己定义的tag标签。简而言之,如果使用taglib,那么你可以设计自己的JSP标记! @6tx5D?  
w##Fpv<m  
一般来说,自定义tag标签主要用于操作隐藏对象、处理html提交表单、访问数据库或其它企业级的服务,诸如邮件和目录操作等等。自定义tag标签的使用者一般都是那些对java编程语言非常精通,而且对数据访问和企业级服务访问都非常熟悉的程序员,对于HTML设计者来说,使得他可以不去关注那些较复杂的商业逻辑,而将精力放在网页设计上。同时,它也将库开发者和库使用者进行合理分工,自定义tag标签将那些重复工作进行封装,从而大大提高了生产力,而且可以使得tag库可用于不同的项目中,完美地体现了软件复用的思想。 D"WkD j"M  
tvH)I px  
在这篇文章中,我们主要讨论: \G"/Myi  
g ` {0I[  
· 什么是自定义tag标签? ZuhT \l  
tO0+~Wm  
· 怎么使用tag标签? h}d7M55#|  
G?g7G,|d  
o 声明要使用的tag库 YS~x-5OE\  
}v!6BU6<Q  
o 找到与之对应的tag处理类 0qZ)$ YKq  
Af%?WZlOq  
o tag标签的类型 FP Mk&  
;K_B,@:'  
· 自定义tag标签 t.TQ@c+,J  
oe<Y,%u"6  
o tag处理类 hh{liS% 10  
d"cfSH;h  
o tag库描述 f_\-y&)+*  
 \X`P W  
o tag标签示例 ^ Q}1&w%  
4~/6d9f  
o 带属性的tag tv{.iM|V c  
Qi}LV"&L  
o 带body的tag ][mc^eI0s|  
lyPXlt  
o 定义了脚本变量的tag S .jjB  
EpfmH `  
o 具有协作关系的tag S ] &->5"  
K|/a]I":  
· 自定义tag标签 +u2Co_FJ&  
;n@C(hG  
o 一个迭代tag的例子 h.^DRR^S  
O o:jP6r  
o 一个模板tag库 E.3}a>f  
7 AiCQWf9  
o tag处理类到底是怎样被调用的? [ b W=>M  
Yp)U'8{h c  
w~&]gyf  
什么是自定义的tag? Ed-gYL^<  
2I<T<hFW]  
一个自定义的tag标签是用户定义的一种JSP标记。当一个含有自定义的tag标签的JSP页面被jsp引擎编译成servlet时,tag标签被转化成了对一个称为tag处理类的对象进行的操作。于是当JSP页面被jsp引擎转化为servlet后,实际上tag标签被转化成为了对tag处理类的操作。 mI0r,Z*+M  
MD)"r>k  
自定义tag标签有很多特色,诸如: D^{:UbN  
( A)wcB  
· 可以在JSP页面中自定义tag标签的属性 *J=ol  
cn0Fz"d  
· 访问JSP页面中的所有对象 "m3Y))a  
r;C\eN  
· 可以动态地修改页面输出 x(`$D  
rZv+K/6*M  
· 彼此这间可以相互通信。你可以先创建一个JavaBeans组件,然后在一个tag中调用此JavaBeans组件,同时可以在另一个tag中调用它。 yDC97#%3u  
,Ai i>D]  
· tag允许相互嵌套,可以在一个JSP页面中完成一些复杂的交互。 8hfh,v5(  
!;gke,fB  
|DD?3#G01  
使用tag标签 >C[1@-]G%7  
gT OMD  
lo:~~l  
本节主要描述怎样在JSP页面中使用tag标签,以及tag标签的不同类型。 c5R{Sl  
yh:,[<q  
要使用tag标签,JSP程序员必须做2件事: cZ>W8{G  
L'Zud,JKg  
· 声明此tag标签的tag库 3c3Z"JV  
3Y-v1.^j  
· 实现此tag标签 nS4S[|w"  
E2IVR]C2^  
声明tag标签所在的tag库 q1Sm#_7  
}D+8K  
如果要使用tag标签,则应用JSP的taglib指示符来指定其tag库(注意:taglib要在在使用此tag标签之前声明) zf~zYZSr  
t] wM_]+  
<%@ taglib uri=”/WEB-INF/tutorial-template.tld” prefix=”tt” %> m-RY{DO+  
Ji[g@#  
uri属性定义了唯一的标签库描述(以下简称TLD),它可以是直接是tld文件名或一个独一无二的名字。prefix是用来区别其它TLD中和本TLD中有重名的tag的一种手段。 g-FZel   
Ak Tw?v'  
TLD必须以.tld作为扩展名,并且存放在当前应用的WEB-INF目录或其子目录下。你可以通过它的文件名直接引用它,也可以通过别的方式间接地引用它。 H\mVK!](D  
%#9~V  
以下taglib指示符直接引用一个TLD: Yk Pt*?,P/  
dO,05?q|  
<%@ taglib uri=”/WEB-INF/tutorial-template.tld” prefix=”tt” %> E+zn\v  
RHVv}N0  
以下的taglib指示符通过一个逻辑名称间接地引用一个TLD: m!60.  
17)M.(qmuP  
<%@ taglib uri=”/tutorial-template” prefix=”tt” %> fm>K4\2  
]F;]<_  
如果是间接引用TLD的话,那你必须还要在web.xml中定义此逻辑名称与tld文件之间的映射,具体做法是在web.xml中加入一个名为taglib的元素: 2hJ3m+N^  
QtQ^"d65  
<taglib> ssITe., ny  
0 0&$SE  
<taglib-uri>/tutorial-template</taglib-uri> R+0"B  
|:+pPh!-  
<taglib-location> i(;-n_:, `  
%n25Uq  
/WEB-INF/tutorial-template.tld r5!M;hU1j  
*^6xt7  
</taglib-location> 03WRj+w  
q&Wwt qc9  
</taglib> X&.$/xaT  
[!? ,TGM}^  
-/c1qLdQ  
实现此tag标签 0t?<6-3`/  
K=TW}ZO  
Z(mn U;9{v  
为了实现tag标签,你有2种方法来存放tag处理类。一、让tag处理类以.class的方式存放于当前应用的WEB-INF/class子目录下,二、如果tag处理类是以JAR包的形式存在的话,那可以放在当前应用的WEB-INF/lib目录下,如果tag处理类要在多个应用中共享,那么它就应放在jsp服务器上的common/lib目录下,对于tomcat来说,就是tomcat/common/lib目录下。 O^weUpe\  
YO$b#  
@^cgq3H'  
tag标签类型 Xl6ZV,1=n7  
cGta4;  
IQ=|Kj9h  
自定义的tag标签遵循XML语法。它有一个开始标记和一个结束标记,有的还有body(即文本节点): ,7jiHF  
*.%)rm  
<tt:tag> Y0O<]2yVx  
y~c[sW   
body P=[x!}.I  
h) PB  
</tt:tag> up3<=u{>  
ysJhP .  
OCO,-(  
一个不带body的tag标签如下: Q EGanpz  
RK:sQWG  
<tt:tag /> Qfu*F}  
2G5!u)  
ku9F N  
简单的tag标签 X/,1]  
>m6,xxTR  
一个没有body和属性的tag标签如下: yn ":!4U1  
SA 4je9H%  
<tt:simple /> 2mU-LQ1WN  
; 9&.QR(  
T.P Z}4  
带属性的tag标签 |ezO@  
mRnzP[7-\)  
ae#HA[\0G  
自定义标签可以有自己的属性。属性一般在开始标记中定义,语法为 attr=”value”。属性的作用相当于自定义标签的一个参数,它影响着tag处理类的行为。你可以在TLD中详细定义它。 Qn)[1v  
1fhK{9#  
你可以用一个String常量给一个属性赋值,也可以通过表达式给它赋值,如<%= ...%>。以struts为例,它的logic:present标签就是用的String常量来给属性赋值: QqK{~I|l  
zHc4e   
<loglic:present parameter = “Clear”> 2a(yR >#  
Ldj^O9p(  
而另一个标签logic:iterate是用表达式来给属性赋值: Xa%&.&V  
$_7d! S"  
<logci:iterate collection=”<%= bookDB.getBooks() %>” %%,hR'+|  
m *bKy;'8  
id=”book” type=”database.BookDetails”> xKLcd+hCZ  
q MdtJ(gq  
xVz -_z  
带body的tag标签 dms:i)L2  
zV(tvt  
一个自定义标签可以包含其它自定义标签、脚本变量、HTML标记或其它内容。 i~Ob( YIH  
[(P[qEY  
在下述例子中,此JSP页面使用了struts的logic:present标签,如果些标签定义了parameter=”Clear”的属性,则将清除购物车的内容,然后打印出一条信息: <\9Ijuq}k  
\ NSw<.  
<logic:present parameter=”Clear”> fRa-bqQ  
RQ)!KlY  
<% cart.clear(); %> "ko?att~  
M3;v3 }z<-  
<font color=”#ff0000” size=”+2”><strong> L5qCv -{  
I;.! hV>E  
你选择了清除购物车! &B7+>Ix,  
?)o4 Kt'h  
</strong></font> t k/K0u  
ny_ kr`$42  
</logic:present> {p*hNi)0  
nK%/tdq  
n.Eoi4jV'  
到底是用属性还是用body来传递信息? {L-aXe{  
a(43]d&  
如上所述,我们既可以通过属性,也可以通过body来传递信息。但一般来说,比较简单的类型,如字符串或简单表达式最好采用属性来传递信息。 Gp3nR<+  
`ToRkk&&>{  
k1Mxsd  
定义脚本变量的tag标签 yw Q!9 \  
Q~Sv2  
所谓脚本变量,是指JSP中可以调用的变量或对象。它可由tag标签产生。以下示例阐述了一个tag标签定义了一个名为tx的由JNDI所定义的事务处理对象。脚本变量可以是ejb对象、事务、数据库连接等等: 3|'#n[3  
JXRf4QmG  
<tt:lookup id=”tx” type=”UserTransaction” name=”java:comp/UserTransaction” /> (zw=qbS&  
{Fp`l\,  
<% tx.begin(); %> z2nUul(2  
PxVI {:Uz  
... 6v2RS  
!%RJC,X  
#9hXZr/8  
具有协作关系的tag标签 #nf%ojh  
QOh w  
自定义tag标签之间可以通过共享对象来实现协作。在下述例子中,标签tag1创建了一个名为obj1的对象,在标签tag2仍可以重复使用obj。 LY88;*:S  
e<O;pM:  
<tt:tag1 attr1=”obj1” value1=”value” /> Fb{`a[&  
HSr"M.k5  
<tt:tag2 attr1=”obj1” /> kSDa\l!W]  
Xm^h5jAr  
在以下这个例子当中,如果外层的tag标签创建了一个对象,那么其内层的所有tag标签都可以使用这个对象。由于这样产生的对象没有一个指定的名字,那么就可以将少重名的冲突。这个例子阐述了一系列协作的嵌套对象。 _Dcc<-.  
sg6w7fp>  
<tt:outerTag> G_,t\  
E_![`9i  
<tt:innerTag /> Va/@#=,q]  
B:A1W{l  
</tt:outerTag> k.=S+#"}  
(|a$N.e&K  
x+*L5$;h  
Tag处理类 X v7U<q  
Puth8$  
gcW{]0%L^  
Tag处理类必须实现Tag接口或BodyTag接口,不过现在一般都流行从TagSupport或BodyTagSupport类中继承,这些类或接口都可以在javax.servlet.jsp.tagext包中找到。 K@j^gF/0B  
c]aK N  
当JSP引擎看到自己的JSP页面中包含有tag标签时,它会调用doStartTag方法来处理tag标签的开头,调用doEndTag方法来处理tag标签的结束。 WP% {{zR$  
d0}%%T  
下表说明不同类型的tag所需要不同的处理过程: DvRA2(M  
_^xh1=Qr}n  
Tag处理类的方法 |p8"9jN@}c  
|!xfIR>=F  
Tag标签类型 [`zbf_RyO  
所调用的方法 !.2CAL  
6Er0o{iI  
基本标签 e2-70UvW^  
doStartTag, doEndTag, release \{M rQ2jd  
+.gf]|  
带属性的标签 sQ>B_Y!  
doStartTag, doEndTag, set/getAttribute1...N, release b!^M}s6  
GOZQ5m -  
带内容的标签  ENYF0wW  
doStartTag, doEndTag, release 6Bq~\b^  
M]4=(Vv+5  
带内容的标签,且内容重复循环 )ZgER[  
doStartTag, doAfterBody, doEndTag, release ,. 6J6{  
}W__ffH  
带内容的标签,且内容与JSP交互 /XW&q)z-Hl  
doStartTag, doEndTag, release, doInitBody, doAfterBody, release 8=n9hLhqo  
F; MF:;mM  
一个tag处理类可以通过javax.servlet.jsp.PageContext来与JSP交互,通过javax.servlet.jsp.PageContext类,tag处理类可以访问JSP中的request、session和application对像。 M8#*zCp{5  
!HdvCYB>  
如果tag标签是互相嵌套的,那内层的tag处理类可以通过它的parent属性来访问上层的tag处理类。 1o;g1Z/  
n2jvXLJq  
一般情况都将所有的tag处理类打成了JAR的包,以便于发布。 r{_B:  
ax72ehL}  
~_l6dDJ  
Tag库描述(简称TLD) i!1ho T$  
_\4`  
D8@n kSP  
Tag库是用xml语言描述的,TLD包括了tag库中所有tag标签的描述,它一般用来被jsp服务器用来校验tag的语法正确性,或者被jsp开发者用来开发新的标签。 EZ%w=  
*793H\  
TLD的文件扩展名必须为.tld,而且必须放在当前WEB应用的WEB-INF目录或其子目录中。 T]Tdx.B  
hr_9;,EPh  
一个TLD的内容的开头必须遵守标准的XML开头,用于描述DTD和xml的版本,例如: OD?y  
$IxU6=ajn  
<?xml version="1.0" encoding="ISO-8859-1" ?> #90[PASx  
jIx8k8  
<!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN" "http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd"> AK@`'$  
TLD必须以<taglib>来作为它的根元素,<taglib>的子元素如下表: m{b ZRkt  
jSwtf  
Ss#@=:"P  
<taglib>的子元素 |P,zGy  
!^)wPmk  
Element `x{.z=xC  
Description Sc4obcw%  
N"Qg\PS_  
tlib-version tT@w%Sz57N  
Tag库的版本 MG7 ?N #  
"wnpiB}  
jsp-version ;t;Y.*&=S  
Tag库所需要的jsp的版本 ? fbgU  
VxkCK02k  
short-name ZR;8r Z](  
助记符,tag的一个别名(可选) z.2r@Psk  
(|0.m8D~D  
uri BR& Aq  
用于确定一个唯一的tag库 sJ))<,e5I  
[K cki+  
display-name AfbB~LlBq  
被可视化工具(诸如Jbuilder)用来显示的名称(可选) }J ei$0x  
mQd4#LJ_  
small-icon _pz,okO[V  
被可视化工具(诸如Jbuilder)用来显示的小图标(可选) AW,v  
v`#j  
large-icon ,:#,}w_HyO  
被可视化工具(诸如Jbuilder)用来显示的大图标(可选) !US8aT  
c;:">NR  
description w(76H^e  
对tag库的描述(可选) ID67?:%r  
/9x{^  
listener v2 29H<  
参见下面listener元素 _ztZ> '  
0>,.c2),  
tag  ]{f^;y8  
参见下面tag 元素 }xAie(  
&8[ZN$Xe"  
Listener元素 [>W"R1/  
!c3```*  
一个tag库可能定义一些类做为它的事件侦听类,这些类在TLD中被称为listener 元素,jsp服务器将会实例化这些侦听类,并且注册它们。Listener元素中有一个叫listener-class的子元素,这个元素的值必须是该侦听类的完整类名。 EMVk:Vt]  
?z2jk  
Tag元素 ?QCmSK=L  
B.89_!/:p  
每个tag元素在tag库中都要指出它的名字、类名、脚本变量、tag的属性。其中脚本变量的值可以直接在TLD中定义或通过tag附加信息的类来取得。每个属性描述了这个属性是否可以省略,它的值是否可以通过<%= …%>这样的JSP语法来获得,以及属性的类型。 V]I:2k5  
C`\9c ej  
每一个tag在TLD中对应一个tag元素,下表是tag元素的子元素: ,HFs.9#&B  
$> "J"IX  
Tag元素的子元素 k: b/Gq`  
S~KS9E~\  
元素名称 v,/[&ASz  
描述 2v0!` &?M{  
~I{EE[F>qL  
name 21r= = H$  
独一无二的元素名 x(L(l=^"  
, N53Iic  
tag-class &4,WG  
Tag标签对应的tag处理类 ?Bo?JMV  
OF c\fW#  
tei-class x^A7'ad0  
javax.servlet.jsp.tagext.TagExtraInfo的子类,用于表达脚本变量(可选) ""co6qo#>  
sX+`wc  
body-content T4mv%zzS  
Tag标签body的类型 AY#wVy  
,4[dLWU  
display-name PK_s#uC  
被可视化工具(诸如Jbuilder)用来显示的名称(可选) otO j^xU  
t/}L36@+  
small-icon 'It?wB W  
被可视化工具(诸如Jbuilder)用来显示的小图标(可选) O~V1Ywfq7^  
A (Bk@;  
large-icon u*2fP]n  
被可视化工具(诸如Jbuilder)用来显示的大图标(可选) kw*)/$5]  
P0^c?s"I  
description 8{dEpV*  
此tag标签的描述 ;HDZ+B  
S}[l*7  
variable 3y99O $EAc  
提供脚本变量的信息(同tei-class)(可选) 2 P=[  
&VDl/qnaL  
attribute oL]mjo=jN  
Tag标签的属性名 \K;op2  
L>dkrr)e  
以下章节介绍对于不同类型的tag,如何具体地实现它们。 74+A+SK[  
( S`6Q  
B`fH^N  
简单的tag 2 nv[1@M  
5F2_xH$5  
*ZaaO^!  
tag处理类 GcT;e5D  
@}Zd (o  
简单的tag处理类必须实现Tag接口的doStartTag和doEndTag方法。当jsp引擎碰到tag标签的开头时,doStartTag被调用,因为简单的tag没有body,所以此方法将返回 SKIP_BODY。当jsp引擎碰到tag标签的结尾时,doEndTag被调用,如果余下的页面还要被计算,那它将返回EVAL_PAGE,否则将会返回SKIP_PAGE。 Gqb])gXpl  
]4`t\YaT  
以下是例子:对于标签 <tt:simple /> ,它的tag处理类实现如下: J! {Al  
mzX;s&N#  
'BY-OA#xJ  
public SimpleTag extends TagSupport ?~J i-{#X  
s=D f `  
{ }Dn^d}?s||  
4S|=/f  
public int doStartTag() throws JspException k;k}qq`d  
e+.\pe\  
{ l4rMk^>>  
ldGojnS  
try{ W^es;5  
C-m*?))go  
pageContext.getOut().print(“Hello.”); `5q ;ssu  
,> n% ~'gb  
}catch(Exception e){ 5Fm av5  
8TE>IPjm  
throw new JspTagException(“SimpleTag: “ + e.getMessage()); {CtR+4KD  
]IZ>2!6r  
} ?s?$d&h  
`9Yn0B.  
return SKIP_BODY; (luKn&826  
w&Y{1rF>  
} .6 3=(o  
3uV4/% U  
public int doEndTag() w7FoL  
oKA&An  
{ r3qf[?3`6  
ySe$4deJ  
return EVAL_PAGE; ]N^*tO  
YuQ~AE'i  
} 7G<t"'  
y+9h~,:A  
} %T!J$a)qf  
?P/AC$:|I  
6BocGo({  
注意:如果tag标签没有内容的话,那必须定义body-content元素为空,例如 9@K.cdRjQ  
.$&Q[r3Lu  
<body-content>empty</body-content> e4`uVq5  
a^t?vv  
H6K`\8/SeN  
带属性的tag标签
评价一下你浏览此帖子的感受

精彩

感动

搞笑

开心

愤怒

无聊

灌水
级别: 终身会员
发帖
3743
铜板
8
人品值
493
贡献值
9
交易币
0
好评度
3746
信誉值
0
金币
0
所在楼道
只看该作者 1 发表于: 2007-01-01
tag处理类 2-'Opu  
At6qtoPRA  
对于tag标签的每个属性,你必须依照JavaBeans规范来定义其属性,以及get和set方法。以struts的logic:present 标签为例, +9XQ[57  
F%d \~Vj  
<logic:present parameter=”Clear”> VsK>6S\T  
80pid[F  
与此相应,此tag处理类应有如下方法和定义: F'JY?  
R@iUCT^$  
XL$* _c <)  
protected String parameter = null; O(z}H}Fv  
$w#r"= )  
public String getParameter() #!2k<Q*5uT  
G8Z4J7^  
{ i3VW1~.8  
Km#pX1]>e  
return this.parameter; *\uM.m0$  
_[K"gu  
} Dg HaOAdU  
3;[DJ5  
public void setParameter(String parameter) b:J(b?  
MZ> 6o5K|  
{ FLZWZ;  
/9pM>Cd*Z  
this.parameter = parameter; $((6=39s  
(ljF{)Ml+=  
} I$t3qd{H&  
_>m-AI4^  
44ed79ly0)  
注意:如果你的属性名为id,而且你的tag处理类是从TagSupport类继承的,那你就不需要定义它的属性和set和get方法,因为他们早已在TagSupport被定义过了。 5O/i3m26  
I 1Sa^7  
Attribute元素 %+)o'nf"U  
k S# CEU7  
对于tag标签的每个属性,你必须定义它是否必须的,它的值是否可以用诸如<%= …%>的表达式来获得,以及它的类型(可选),如果不指定它的类型,那就默认为是java.lang.String类型。如果rtexprvalue元素被定义为true或yes,那么在type元素中就定义了attribute的返回类型。 )B# ,  
h#r^teui)  
<attribute> \2 y5_;O  
S=`+Ryc  
<name>attr1</name> a:TvWzX,  
Kl{>jr8B3  
<required>true|false|yes|no</required> 6 K` c/)  
`d]IX^;  
<rtexprvalue>true|false|yes|no</rtexprvalue> JAjmrX  
'XrRhF (  
<type>attribute的返回类型(只用当rtexprvalue为真是才有效)</type> 4+;$7"fJ  
4mjgt<`  
</attribute> Y-mK+1 2  
{c?JuV4q?  
如果tag的某个属性不是必须的,那tag处理类会自动提供一个缺省值。 lbdTQ6R  
H9)m^ *  
例如,在logic:present这个tag标签中定义了一个属性叫parameter,但它不是必须的,而且它可以被诸如<%= …%>的表达式来赋值。 O,2~"~kF  
i':i_kU  
<tag> gi/@ j  
B+d<F[ |  
<name>present</name> F>je4S;  
a]Eg!Q  
<tag-class>org.apache.struts.taglib.logic.PresentTag</tag-class> A>`945|  
51C2u)HE  
<body-content>JSP</body-content> X?:o;wB  
IP`6bMd  
6qWdd&1  
\c v?^AI  
<attribute> 2&'|Eqk  
7uorQfR?  
<name>parameter</name> |BT MJ:B  
vbx6I>\Y  
<required>false</required> u]-_<YZ'B  
1n5(S<T  
<rtexprvalue>true</rtexprvalue> @`opDu!  
:2 >hoAJJ  
</attribute> 0Sq][W=  
'>$EOg"  
X,aYK;q%z  
\0l>q ,  
</tag> U[L9*=P;  
 VGHWNMT  
s>k Uh  
属性元素的校验 7|\@zQ h   
`\`>0hlu  
*L6PLe  
有关于tag标签的有效值可以从tag库的说明文档中获得,当JSP页面被编译时,jsp引擎会强制性地参照TLD中定义的规则进行检查。 n79QJl/  
;8WZx  
还有一个方法也可以进行属性元素的校验,就是先继承类TagExtraInfo,然后调用它的isValid方法。这个类同时也起到提供tag中定义的脚本变量信息的作用。 T{qTj6I  
H1GRMDNXOA  
IsValid方法通过TagData对象来传递属性信息,它包括着tag的所有的属性名-值的信息。由于校验发生在运行时刻,因此这个属性的值将被赋值为TagData.REQUEST_TIME_VALUE。 Jj~EiA  
 T9)nQ[  
例如tag标签<tt:twa attr1=”value1” />在TLD中定义如下: &cWjE x  
O%g $9-?F0  
<attribute> 1g# #sSa6  
b`yZ|j'ikd  
<name>attr1</name> W?yd#j  
b*a2,MiM  
<required>true</required> 9a\nszwa  
|(m oWY=  
<rtexprvalue>true</rtexprvalue> IK,|5]*Ar  
gqXS~K9t  
</attribute> 6S6f\gAM  
<FMq>d$\  
^ -FX  
这个定义说明了attr1能在运行期间被赋值。 yR{x}DbG  
b" xmqWa  
以下的isValid方法检查attr1属性的值是否属于Boolean类型。注意由于attr1能在运行刻被赋值,那么isValid方法必须检查tag用户是否对此tag进行了运行时刻赋值。 CT0l!J~5m~  
C%*k.$#r!  
Mb3}7@/[  
Public class TwaTEI extends TagExtraInfo Om{l>24i.\  
k#[F`  
{ (b?{xf'G  
oH0X<'  
public boolean isValid(Tagdata data) 43?^7_l-  
_&K  
{ |KB0P@=a  
:m86 hBE.  
Object o = data.getAttribute(“attr1”); D=:04V}2+  
yC 77c=  
If(o != null && o != TagData.REQUEST_TIME_VALUE) UnVm1ZWZ  
@(P=Eh  
{ !fBF|*/  
t8^m`W  
if( ( (String)o).toLowerCase().equals(“true”) || Y(cN}44  
+&zYZA8v  
((String)o).toLowerCase().equals(“false”) ) yc|VJ2R*  
 ^p n(=4  
return true; k = ?h~n0M  
WI]o cF  
else ^[%%r3"$C  
V8eB$in  
return false; S'oGt&Z<  
Z/rP"|EuQ  
} 1B),A~Ip  
tXJU vish  
else BCe_@  
aP'"G^F   
return true; ARcv;H 5  
w9 w%&{j  
} u77E! z4Uz  
XLMb=T~S  
} s1|/S\   
q+B&orp  
!`!| Zw  
带body的tag ~Lc066bLeq  
Y+K|1r  
cYXM__  
tag处理类 /1?R?N2>0  
@ HZKc\1  
cRX~z  
如果tag标签含有内容,那处理方式会略微有些不同,而且还要视tag处理类是否要与body交互的情况而定。如果要与body交互,那我们认为tag处理类要可能要对body进行操作。 lL]y~u  
4&/j|9=X  
Tag处理类不与body交互 ]|<w\\^A  
Xl@cHO=i  
如果tag处理类不与body交互,tag处理类应该实现Tag接口或从TagSupport中派生,如果body要被计算,那么doStartTag方法应返回 EVAL_BODY_INCLUDE,否则应返回SKIP_BODY。 AoA!q>  
WyP W*  
如果tag处理类要对body反复运算,则它应该实现IterationTag或从TagSupport中派生。如果tag处理类认为body还未计算完的话,那它的doStartTag方法和doAfterBody方法必须返回EVAL_BODY_AGAIN。 eY{+~|KZ  
~=R SKyzt  
Tag处理类与body交互 > iE!m  
}I`a`0/  
如果tag处理类与body交互,那tag处理类应实现BodyTag接口或从BodyTagSupport中派生。这种tag处理类一般要实现doInitBody和doAfterBody方法。 iNwqF0  
<b/~.$a'  
Body允许一些方法来读写它的内容。Tag处理类可以调用body内容的getString或getReader方法来从body中提取信息,也可用 writeOut(out) 方法来将body的内容写入到out对象中。其中out对象通过tag处理类的getPreviousOut方法来获得。 FI"`DMb}  
s1?[7yC  
如果body的内容需要被计算,那么doStartTag方法必须返回EVAL_BODY_BUFFERED,否则,它将返回 SKIP_BODY。 p4p@^@<>X  
~b {Gz6u>  
doInitBody 方法 ;[RZ0Uy=  
nx0K$ Ptq  
此方法在body内容已经设好,但未被计算之前被调用。你可以根据不同的body内容来制定初始化策略。 +cU>k}  
qRbf2;  
doAfterBody方法 h*u`X>!!  
;gC|  
此方法在body内容已被计算后进行调用。 fwzb!"!.@  
AkOO )0  
和doStartTag方法一样,doAfterBody方法返回一个指示符指示是否要继续计算body,如果要继续计算,则doAfterBody应返回EVAL_BODY_BUFFERED,否则,它应返回SKIP_BODY。 \.mI  
<AJ97MLcc  
release 方法 tGB@$UmfU  
HHqwq.zIy  
tag处理类调用此方法将它的状态重置为初始状态,并释放所有的私有资源。 Gycm,Cy  
dg4vc][  
[]s^   
以下的例子读取body的内容(其中含有一条sql语句),然后将它传递给一个对象,让它进行查询。由于此处body不须重新计算,所以doAfterBody会返回SKIP_BODY。 l }XU 59  
Z$J#|  
dL|+d:v  
Public class QueryTag extends BodyTagSupport jY_T/233d  
!%dN<%Ah  
{ ?W E  
m|OO,gR  
public int doAfterBody() throws JspTagException _HhbIU  
a1Qv@p^._b  
{ xeGb?DPu  
\c^45<G2qA  
BodyContent bc = getBodyContent(); y^o@"IYu3  
v9T_&  
//将body的内容以字符串的格式提取出来 v@#b}N0n  
3]?#he  
String query = bc.getString(); %Qk/_ R1   
<V>dM4Mkr  
//清除body UwC=1g U  
_#vrb;.+  
bc.clearBody(); Xy%p"b<  
imiR/V>N  
try{ G\T fL^A  
^] kF{ o?  
Statement stmt = connection.createStatement(); WOh|U4vt  
)& u5IA(  
Result result = stmt.executeQuery(query); -(K9s!C!.  
=/\:>+p^.y  
}catch(SQLException e){ QNDHOo>v  
Hr$QLtr  
throw new JspTagException(“queryTag: “ + e.getMessage() ); "Ky; a?Y  
h,"4SSL  
return SKIP_BODY; ^eoLAL  
s=[h?kB  
} F`9]=T0  
U!Ek'  
} H:"ma S\I  
=N 5z@;!  
1!>Jpi0  
body-content元素 2h%z ("3/  
@O[5M2|r  
N]RZbzK_5G  
由于tag可能会有body,你必须用body-content元素来指定body内容的类型: =Fdg/X1  
]5%/3P,/  
<body-content>JSP|tagdependent</body-content> }- Wa`t7U  
"*})3['n  
;t+ub8  
如果body的内容是定制的或内部的tag、脚本元素、或HTML广本,则归类为JSP类型。其他的类型,比如上面代码所述的?D?D将sql statement类传给 query tag的这种类型应该标为tagdependent。 jbR0%X2  
E\C9|1)  
注意:实际上body-content的值并不影响tag处理类对body内容的处理,它仅仅是被tag编辑工具用来描述此body的内容。 K(q-?n`<  
*YlV-C<}W"  
>$2V%};  
用tags定义脚本变量 "le>_Ze_>|  
p0pWzwTG3  
@}kv-*  
tag处理类 VcoOeAKL  
*_?dVhxf  
0:b2(^]bg  
tag处理类负责创建或设置页面中定义的脚本变量,用pageContext.setAttribute(name,value,scope)或pageContext.setAttribute(name,value)方法来实现。一般来说,tag处理类通过脚本变量的名称来获取它,脚本变量的名称一般可用get方法来获得。 RVeEkv[qp  
_/O25% l  
如果脚本变量的值依赖于tag处理类中的上下文中某一对象,那它可用pageContext.getAttribute(name,scope)方法来找到那个对象。一般的处理过程是tag处理类先找到脚本变量,再对其进行处理,然后用pageContext.setAttribute(name,object)的方法来设置它的新值。 +k`!QM>e-  
+E1h#cc)  
对象的生存周期(scope)如下表: <vwkjCA`  
Onwp-!!.  
对象的生存周期表 ~,*b }O  
@'GGm#<   
名字 ]7e =fM9V;  
可访问范围 hqRw^2F  
生存周期 6"}?.E$  
}3?n~s\)6f  
page @lvyDu6e  
当前页面 "Y\_TtY  
一直有效,除非页面向客户提交响应或重定向到一个新页面 #UbF9})q  
cH>%r^G\  
request R+CM`4CD  
当前页面或当前页面重定向到的页面 O|w J)  
一直有效,除非页面向客户提交响应 KIWe@e  
%dY<=x#b  
session xNbPsoK  
当前页面或在同一浏览器窗口中的页面 yiO. z  
一直有效,除非关闭当前浏览器、超时、网络故障 o^ XtU5SVq  
[]D@Q+1  
application 2p " WTd  
整个web应用程序的所有请求 p/h Rk<K6  
一直有效,除非发生网络故障、服务器故障 5L!y-3  
tToTxf~  
提供关于脚本变量的信息 7nuU^wc  
`]W| 8M  
以下示例定义了一个名为“book”的脚本变量,用来访问程序中关于书的信息: |6< p(i7  
L`24 ?Y{  
<bean:define id=”book” name=”bookDB” property=”bookDetails” type=”database.BookDetails” /> J_;o|gqX  
? YG)I;(  
<font color=”red” size=”+2” > o]opdw  
_]:z \TDn  
<%= messages.getString(“CartRemoved”) %> #_u~/jhX  
Hhh0T>gi  
<strong><jsp:getProperty name=”book” property=”title” /></strong> KRA/MQ^7~U  
_F`lq_C  
</font> Ih"XV  
' MxrQ;|S  
当包含此tag的JSP页面被编译时,jsp引擎会自动生成关于此book的同步的代码(同步可以避免几个客户同时访问此book时造成的冲突),要生成同步代码,jsp引擎需要知道此脚本变量的如下信息: V 0rZz  
y`=]T>X&x  
· 脚本变量名称  E@b(1@  
ctGL-kp  
· 脚本变量所属的类 GN2Sn` ;  
lg&t8FHa;  
· 此脚本变量是否引用了一个新的或已存在的对象 &c,kQo+pA  
VzVc37 Z>6  
· 此脚本变量的有效性 b1( $R[  
7"C$pm6  
有两种办法可以向jsp引擎提供关于脚本变量的信息:在TLD中定义variable子元素,或用tei-class子元素定义一个额外tag信息类。用variable最简单,但可能降低了一些灵活性。 j}C}:\-fY  
Ct>GYk$  
Variable元素 UNBH  
mrjswF27$o  
Variable元素有如下子元素: V=*wKuB  
_D+J!f^  
· name-given ?D?D 给出的名字,是一个常量 X93!bB  
r! MWbFw|X  
· name-from-attribute?D?D 属性名,在编译时给出的属性名 N}t 2Nu-  
\7'+h5a  
name-given或name-from-attribute两者必须选一,但以下子元素是可选的: 0ik7v<:  
9_5ow  
· variable-class?D?D变量的类型,缺省为java.lang.String。 ruld B,n  
KGFv"u{  
· declare?D?D此脚本变量是否引用了一个新对象,缺省为True。 ;4pYK@9w_  
q0zr E5  
· scope?D?D脚本变量的范围,缺省为NESTED。下表描述了scope的几种类型: sjV!5Z  
\vO,E e~#W  
脚本变量的有效范围 uu>Pkfo  
@8I4[TE  
;N?]eM}yf  
有效性 (R("H/6xs  
方法 53n^3M,qK  
;67x0)kn  
NESTED LBZ+GB  
在tag标签的开始和结束之间 !/]WrGqbS  
如果tag处理类实现BodyTag接口,则在doInitBody和doAfterBody中调用,否则在doStartTag中调用 |mw.qI|  
=UfsL%  
AT_BEGIN XSyHk"g`  
从tag标签的开始一直到页面结束 m+T;O/lG0{  
如果tag处理类实现BodyTag接口,则在doInitBody、doAfterBody和doEndTag中调用,否则在doStartTag和doEndTag中调用 e-EUf  
D1=((`v '  
AT_END ys kO  
从tag标签的结束一直到页面结束 Z '7  
在doEndTag中调用 P`cq H(   
VG ;kPzze  
以struts为例,它的bean:define标签的实现遵循JSP1.1规范,此规范要求使用额外tag信息类来定义脚本变量。Variable元素是JSP1.2规范中加入的。以bean:define标签为例,你可以定义如下variable元素: )`(p9@,V  
#$8% w  
<tag> ", KCCis  
$cU!m(SILQ  
<variable> $arK(  
|M, iM]  
<name-from-attribute>id</name-from-attribute> QvKh,rBFVG  
7V!*NBsl  
<variable-class>database.BookDetails</variable-class> VL` z[|e @  
#ZrHsf P  
<declare>true</declare> ) iN/ua  
>E{";C)  
<scope>AT_BEGIN</scope> DBr ZzA  
lSVp%0jR  
</variable> -^hWM}F  
EZ`te0[  
</tag> BdH-9n~,  
3!|;iJRH  
额外tag信息类 ud'-;W  
"4{LN}`  
如果要定义一个额外tag信息类,你要继承javax.servlet.jsp.TagExtraInfo类。一个TagExtraInfo类必须实现getVariableInfo方法,此方法返回一个叫VariableInfo的数组类,它包括如下信息: ^Dn D>h@q  
j~jV'f.:H  
· 变量名 =*c7i]@}  
.7avpOfz  
· 变量所属类名 #PH~1`vl  
IS&ZqE(`e  
· 此变量是否引用了一个新对象 NUWDc]@J*  
=k^Y?.  
· 此变量的有效范围 p o2!  
%D%8^Zd_  
jsp引擎将一个名为data的参数传给getVariableInfo方法,data中包括tag标签中的所有“属性名?D?D属性值”对。它可以用来向VariableInfo对象提供脚本变量的名字和类名。 a C\MJ9  
OX?\<),  
以struts为例,它在bean:define标签中定义了一个名为DefineTei的额外tag信息类,用来向脚本变量提供信息。由于脚本变量的名称(book)和类名(database.BookDetails)是通过tag标签的属性来传递的,它们一般定义在VariableInfo的构建代码中,并且可用data.getAttributeString方法来得到这些信息。如果要允许book脚本变量能在从tag开始直到整个JSP页面结束的范围内都可用的话,那它的范围应设为AT_BEGIN。如下所示: n%.7h3  
9Hb6nm  
'R'*kxf  
public class DefineTei extends TagExtraInfo V8C:"UZ;  
pUQ/03dp  
{ Ch|jtVeuyJ  
':3 pq2{  
public VariableInfo[] getVariableInfo(TagData data) {YAJBIvHV  
jN;@=COi  
{ lJ}_G>GJ  
DpvI[r//'*  
String type = data.getAttributeString(“type”); '}Z~JYa0  
sHt].gZ  
If( type == null) y[)>yq y  
CR"|^{G  
type = “java.lang.Object”; i%#+\F.&  
[ 0KlC1=  
return new VariableInfo[] { xy/`ZS2WPq  
{E9+WFz5  
new VariableInfo(data.getAttributeString(“id”), mpU$ +  
,*&:2o_r  
type, 4~Vx3gEV:  
KWowN;  
true, e478U$  
>>t@}F)  
VariableInfo.AT_BEGIN) Eg#K.5hJ  
wnEyl[ac  
};  8pIP  
YQ9'0F[l  
} i@)i$i4  
75f"'nJ)  
} d iL +:H  
-pf}  
59Xi3KY  
注意:关于额外tag信息类的类名必须要在TLD中的tag标签下的tei-class子元素中定义。因此,DefineTei的tei-class中的定义看起来如下: s E2D#D  
<tei-class> 8 D3OOab  
org.apache.struts.taglib.bean.DefineTagTei mS$j?>m  
</tei-class> tl,.fjZn  
=[cS0Sy  
bLij7K 2H  
具有协作关系的tag 7Bzq,2s  
pfA|I*`XV  
4:$4u@   
tag通过共享对象来进行协作,JSP技术支持2种方式的对象共享。 QwJV S(Gs4  
N kb|Fd/s  
第一种方法是使用pageContext对象进行对象的共享(可支持JSP页面和tag处理类之间的共享),如果在一个tag处理类中要调用由另一个tag处理类创建的对象,可调用pageContext.getAttribute(name, scope)方法。 G'Q-An%z  
fTS5 yb%  
第二各方式的共享是对于tag之间有嵌套关系而言的,外层的tag所创建的对象对于内层的tag来说是可以共用的。这种形式的共享的好处是减少了可能存在的重名冲突。 JQ8fdP A  
r@h5w_9  
要访问一个嵌套tag创建的对象,tag处理类必须先找到此嵌套tag对象,可用TagSupport的静态方法 TagSupport.findAncestorWithClass(from, class)或TagSupport.getParent方法。前者在当不确定此tag是否为嵌套tag对象时使用。一旦它的父类被找到,它就能访问其所有动态或静态创建的对象。静态创建的对象是父类的成员,而动态创建的对象可能是父类的私有对象。诸如此类的对象可以用tag处理类的setValue方法来保存,用getValue方法来获得。 q<[P6}.  
zZPuha8  
下例阐述了以上两种共享对象的方法。在这个例子当中,一个查询tag检查一个名为connection的属性名是否在doStartTag中被设置。如果connection属性被设置,tag处理类从pageContext中得到这个connection对象。否则,此tag处理类先找到它的父tag处理类,然后从它的父tag处理类中找到connection对象。 e6R}0w~G  
_~IR6dKE  
X0bN3N  
public class QueryTag extends BodyTagSupport R_W+Ylob  
n'wU;!W9  
{ GK )?YM  
8_BV:o9kL  
private String connectionId; J>wt (] y  
NO "xL,  
public int doStartTag() throws JspException 9YIM'q>`v  
:~e>Ob[,"  
{ R]c+?4J  
I5 o)_nc  
String cid = getConnection(); p Dx1z|@z  
&=Ar  
if(cid != null) :mh_G  
m4hX 'F  
{ E4`N-3  
-LK B$   
//存在一个connection id,使用它。 TyD4|| %  
!"HO]3-o  
connection = (Connection) pageContext.getAttribute(cid); J*yf2&lI5  
R]}}$R`j  
} ]i&6c  
dt \TQJc~  
else ck ]Do!h  
<k eVrCR  
{ nhB1D-  
gp};D  
ConnectionTag ancestorTag = (ConnectionTag)findAncestorWithClass(this, 8;b( 0^  
m ,* QP*  
ConnectionTag.class); \\PjKAsh  
$UMFNjL  
if(ancestorTag == null) Ygm`ZA y  
1-%fo~!l  
{ a,@]8r-"  
>:AARx%  
throw new JspTagException(“一个没有connection属性的查询标签必须被一个connection标记嵌套。”); YIn',]p:  
;(f) &Yom  
} .*@;@06?  
 8{wwd:6  
connection = ancestorTag.getConnection(); 9oRy)_5Z(=  
/[a~3^Gs^  
} Tzt8h\Q^z  
-[ *,^Ti`  
} SN9kFFIPb=  
m'Amli@[  
} 3EV;LH L  
k$R~R-'  
~ Sg5:T3  
b*;Si7-  
此查询标签在JSP页面中的调用形式可以从以下2种定义中任选一种: w j*,U~syB  
Jj>?GAir  
NO7J!k?  
<tt:connection id=”con01” ...> ... </tt:connection> +6sy-<ZL:  
1t:Q_j0Ym  
<tt:query id=”balances” connection=”con01” > $*^kY;  
?Nup1 !D  
SELECT account, balance FROM acct_table 2KB\1&N  
!*s?B L  
where customer_num = <%= request.getCustno() %> iqC|G/  
_7Rr=_1}  
</tt:query> `> 7; !  
chcbd y>C  
14Xqn8uOW  
6_J$UBT  
^Ew]uN>,  
<tt:connection ...> 8UXjm_B^'  
@)UZ@ ~R  
<x:query id=”balances”> ^ssK   
lW+\j3?Z$  
SELECT account, balance FROM acct_table :}Xll#.,m  
O!m vJD  
where customer_num = <%= request.getCustno() %> 5QW=&zI`=  
`_BNy=`s*  
</x:query> fL_4uC i\  
#^`4DhQ/ 1  
</tt:connection> w,.+IV$Kk  
"W=AB&  
NaPt"G  
与此同时,在TLD中必须指定connection属性为可选的,定义如下: ;9[fonk  
<LmIK  
<tag> O}+.U<V  
NO~*T?&  
... Uddr~2%(  
p31NIf `  
<attribute> VvvRRP^q  
4H,`]B8(D  
<name>connection</name> n(b(yXYm]  
!9u|fnC9  
<required>false</required> J4QXz[dG  
931bA&SL=/  
</attribute> %=n!Em(  
`Bo*{}E  
</tag>
描述
快速回复

您目前还是游客,请 登录注册
欢迎提供真实交流,考虑发帖者的感受
认证码:
验证问题:
10+5=?,请输入中文答案:十五