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

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

级别: 终身会员
发帖
3743
铜板
8
人品值
493
贡献值
9
交易币
0
好评度
3746
信誉值
0
金币
0
所在楼道
标准的JSP 标记可以调用JavaBeans组件或者执行客户的请求,这大大降低了JSP开发的复杂度和维护量。JSP技术也允许你自定义taglib,其实换句话说,taglib可以看成是对JSP标记的一种扩展,正如xml是对html的一种扩展一样。taglib通常定义在tag标签库中,这种标签库存放着你自己定义的tag标签。简而言之,如果使用taglib,那么你可以设计自己的JSP标记! m"X0Owx  
u <D&RT  
一般来说,自定义tag标签主要用于操作隐藏对象、处理html提交表单、访问数据库或其它企业级的服务,诸如邮件和目录操作等等。自定义tag标签的使用者一般都是那些对java编程语言非常精通,而且对数据访问和企业级服务访问都非常熟悉的程序员,对于HTML设计者来说,使得他可以不去关注那些较复杂的商业逻辑,而将精力放在网页设计上。同时,它也将库开发者和库使用者进行合理分工,自定义tag标签将那些重复工作进行封装,从而大大提高了生产力,而且可以使得tag库可用于不同的项目中,完美地体现了软件复用的思想。 I zM=?,`  
lmQ!q>N  
在这篇文章中,我们主要讨论: y3c]zDjV  
(#X/sZQh  
· 什么是自定义tag标签? -O^b  
:7@"EW  
· 怎么使用tag标签? =p N?h<dc  
Xv9kJ  
o 声明要使用的tag库 o/2\8   
+, p  
o 找到与之对应的tag处理类 %* vYX0W"  
9 qqy(H  
o tag标签的类型 Xe:B*  
s80:.B  
· 自定义tag标签 ofj7$se  
Q?I)1][ !"  
o tag处理类 !94& Uk(O  
,#`gwtFG  
o tag库描述 e7>)Z  
9IN =m 5  
o tag标签示例 .A&Ey5  
t`Mm  
o 带属性的tag xXp\U'Ad~~  
$6Nm`[V  
o 带body的tag *{)[:;  
C W7E2 ^P$  
o 定义了脚本变量的tag FWq 6e,  
gE`G3kgn{  
o 具有协作关系的tag L7wl3zG  
FOM~Uj  
· 自定义tag标签 v}+axu/?  
}sd-X`lZ  
o 一个迭代tag的例子 j K$4G.x  
Y^2`)':  
o 一个模板tag库 `0=0IPVd  
?G* XZ0u~  
o tag处理类到底是怎样被调用的? Jr%u[d>  
42Tjbten_u  
7h&$^  
什么是自定义的tag? Zo-E0[9  
os/~6  
一个自定义的tag标签是用户定义的一种JSP标记。当一个含有自定义的tag标签的JSP页面被jsp引擎编译成servlet时,tag标签被转化成了对一个称为tag处理类的对象进行的操作。于是当JSP页面被jsp引擎转化为servlet后,实际上tag标签被转化成为了对tag处理类的操作。 n-}:D<\7  
"5synfO  
自定义tag标签有很多特色,诸如: ~Rs|W;  
F,11 \j  
· 可以在JSP页面中自定义tag标签的属性 0u'4kF!P!  
=yz#L@\!  
· 访问JSP页面中的所有对象 \&&kUpI  
DcQ^V4_  
· 可以动态地修改页面输出 uQ vW@Tt  
zb?wl fT  
· 彼此这间可以相互通信。你可以先创建一个JavaBeans组件,然后在一个tag中调用此JavaBeans组件,同时可以在另一个tag中调用它。 |n}W^}S5  
HvKueTQ  
· tag允许相互嵌套,可以在一个JSP页面中完成一些复杂的交互。 l(v$+  
GH7{_@pv8  
`JG~%0Z?}  
使用tag标签  +h9U V  
hIXGfvUy  
5y(irbk7  
本节主要描述怎样在JSP页面中使用tag标签,以及tag标签的不同类型。 = EyxM  
= <O{t#]  
要使用tag标签,JSP程序员必须做2件事: >QE^KtZ  
3${?!OC  
· 声明此tag标签的tag库 V~Tjz%<  
V*=cNj  
· 实现此tag标签 Gqk"%irZ  
_a+0LTo".  
声明tag标签所在的tag库 Eh!%Ne O  
9(ZzwkD'>  
如果要使用tag标签,则应用JSP的taglib指示符来指定其tag库(注意:taglib要在在使用此tag标签之前声明) y z9`1R2c  
6KnD(im  
<%@ taglib uri=”/WEB-INF/tutorial-template.tld” prefix=”tt” %> ]WWre},  
,RCjfX a  
uri属性定义了唯一的标签库描述(以下简称TLD),它可以是直接是tld文件名或一个独一无二的名字。prefix是用来区别其它TLD中和本TLD中有重名的tag的一种手段。 cuq7eMG6z  
+(/XMx}a  
TLD必须以.tld作为扩展名,并且存放在当前应用的WEB-INF目录或其子目录下。你可以通过它的文件名直接引用它,也可以通过别的方式间接地引用它。 ?9H7Twi+T  
rHT8a^MO  
以下taglib指示符直接引用一个TLD: 6ynQCD  
@ RTQJ+ms  
<%@ taglib uri=”/WEB-INF/tutorial-template.tld” prefix=”tt” %> J:?t.c~$o  
v0"|J3  
以下的taglib指示符通过一个逻辑名称间接地引用一个TLD: tborRi)  
Mky$#SI11  
<%@ taglib uri=”/tutorial-template” prefix=”tt” %> Ef%8+_  
^ b-H  
如果是间接引用TLD的话,那你必须还要在web.xml中定义此逻辑名称与tld文件之间的映射,具体做法是在web.xml中加入一个名为taglib的元素: Pk8L- [&v  
sn5N9=\+T  
<taglib> ^>%=/RX  
&Ei dc .  
<taglib-uri>/tutorial-template</taglib-uri> *XniF~M  
y^;qT_)#  
<taglib-location> RVI],O  
P%ZWm=lg  
/WEB-INF/tutorial-template.tld rt^z#2$  
~gI%   
</taglib-location> .8b 4  
<i]%T~\Af)  
</taglib> V9Au\  
C>T6{$xkC  
Z]w_2- -  
实现此tag标签 O])/kS`  
et9 c<'  
Tw!x*  
为了实现tag标签,你有2种方法来存放tag处理类。一、让tag处理类以.class的方式存放于当前应用的WEB-INF/class子目录下,二、如果tag处理类是以JAR包的形式存在的话,那可以放在当前应用的WEB-INF/lib目录下,如果tag处理类要在多个应用中共享,那么它就应放在jsp服务器上的common/lib目录下,对于tomcat来说,就是tomcat/common/lib目录下。 AQQj]7Y  
]"/ *7NM  
Z NCq /  
tag标签类型 ya:H{#%6  
{F@;45)o  
.*Hv^_  
自定义的tag标签遵循XML语法。它有一个开始标记和一个结束标记,有的还有body(即文本节点): tCc}}2bC&  
Y Nq<%i!>  
<tt:tag> + s[(CI.b  
B5I(ai7<M  
body cq[}>5*k  
?7G[`@^Y  
</tt:tag> 'kx{0J?  
KJ9~"v  
QQ!,W':  
一个不带body的tag标签如下: qZ1fQN1yG  
E+~~d6nB  
<tt:tag /> 2B!nLL Cp+  
i^QcW!X&  
brTB /(E  
简单的tag标签 Q|xPm:  
pkk0?$l ",  
一个没有body和属性的tag标签如下: pn =S%Qf]  
{ld([  
<tt:simple /> l;|1C[V  
v WhtClJ3  
bjvi`jyL3k  
带属性的tag标签 c69C=WQ  
DD~8:\QD  
ol:_2G2xQ  
自定义标签可以有自己的属性。属性一般在开始标记中定义,语法为 attr=”value”。属性的作用相当于自定义标签的一个参数,它影响着tag处理类的行为。你可以在TLD中详细定义它。 #vDe/o+=  
0e}L Z,9e  
你可以用一个String常量给一个属性赋值,也可以通过表达式给它赋值,如<%= ...%>。以struts为例,它的logic:present标签就是用的String常量来给属性赋值: pkxW19h*0  
DAvAozM  
<loglic:present parameter = “Clear”> :^ cA\2=  
N86Hn]#  
而另一个标签logic:iterate是用表达式来给属性赋值: W0nRUAo[  
HX=`kkX  
<logci:iterate collection=”<%= bookDB.getBooks() %>” 6'Lij&,f?{  
FvvF4 ,e5  
id=”book” type=”database.BookDetails”> ;XTP^W!6f  
Zd5fr c$  
G4=v2_]  
带body的tag标签 ;ga~ae=Fg  
`Y(/G"]  
一个自定义标签可以包含其它自定义标签、脚本变量、HTML标记或其它内容。 "urQUpF  
7,(:vjIXd  
在下述例子中,此JSP页面使用了struts的logic:present标签,如果些标签定义了parameter=”Clear”的属性,则将清除购物车的内容,然后打印出一条信息: n\P{Mc  
7%?A0%>6G  
<logic:present parameter=”Clear”> w-\GrxlbX  
|9(uiWf  
<% cart.clear(); %> + 1cK (Si  
Tlf G"HzZ%  
<font color=”#ff0000” size=”+2”><strong> /[us;=CM  
IRcZyry  
你选择了清除购物车! zx\?cF  
>Vg<J~[g  
</strong></font> py,z7_Nuh  
\Mi y+<8$  
</logic:present> un..UU4  
P [gqv3V  
/gPn2e;  
到底是用属性还是用body来传递信息? Jj([O2Eq$  
J.*dA j  
如上所述,我们既可以通过属性,也可以通过body来传递信息。但一般来说,比较简单的类型,如字符串或简单表达式最好采用属性来传递信息。 ^(a%B  
$_<,bC1[  
,H.q%!{h_  
定义脚本变量的tag标签 =m1B1St2  
Mb+cXdZb  
所谓脚本变量,是指JSP中可以调用的变量或对象。它可由tag标签产生。以下示例阐述了一个tag标签定义了一个名为tx的由JNDI所定义的事务处理对象。脚本变量可以是ejb对象、事务、数据库连接等等: %pV/(/Q  
oU{m\r  
<tt:lookup id=”tx” type=”UserTransaction” name=”java:comp/UserTransaction” />  G#K=n  
4m%_#J{  
<% tx.begin(); %> F7L+bv   
P~=|R9 t  
... CxwZ$0  
!R4`ihi1  
bl:.D~@  
具有协作关系的tag标签 EAx@a%  
mO]>(^c  
自定义tag标签之间可以通过共享对象来实现协作。在下述例子中,标签tag1创建了一个名为obj1的对象,在标签tag2仍可以重复使用obj。 Bm.%bA>  
LiiQ;x  
<tt:tag1 attr1=”obj1” value1=”value” /> M"5,8Q`PkI  
7X1T9'j I2  
<tt:tag2 attr1=”obj1” /> HP&+ 8  
_M`--.{\O[  
在以下这个例子当中,如果外层的tag标签创建了一个对象,那么其内层的所有tag标签都可以使用这个对象。由于这样产生的对象没有一个指定的名字,那么就可以将少重名的冲突。这个例子阐述了一系列协作的嵌套对象。 #mCL) [  
%^4CSh  
<tt:outerTag> U4C 9<h&  
{E9+WFz5  
<tt:innerTag /> <66X Xh.  
(3 Two}  
</tt:outerTag> :toh0oB[  
i]YV {  
t4zkt!`B  
Tag处理类 Cz\e w B  
j7$e28|_n  
(a.z9nqGA  
Tag处理类必须实现Tag接口或BodyTag接口,不过现在一般都流行从TagSupport或BodyTagSupport类中继承,这些类或接口都可以在javax.servlet.jsp.tagext包中找到。 =S+wCN  
wsZF;8ut  
当JSP引擎看到自己的JSP页面中包含有tag标签时,它会调用doStartTag方法来处理tag标签的开头,调用doEndTag方法来处理tag标签的结束。 p.v0D:@&  
|,gc_G  
下表说明不同类型的tag所需要不同的处理过程: j,lT>/  
Zw5Ni Xj  
Tag处理类的方法 n 22zq6m  
"U>JM@0DNm  
Tag标签类型 &5fJPv &  
所调用的方法 Pq;U &,  
\r^qL^  
基本标签 |:nOp(A\*  
doStartTag, doEndTag, release q<[P6}.  
8<S~Z:JK  
带属性的标签 oTU!R ,  
doStartTag, doEndTag, set/getAttribute1...N, release nm5cpnNl  
rzhWw-GY  
带内容的标签 #%iDT6  
doStartTag, doEndTag, release W=^#v  
`w#Oih!6A|  
带内容的标签,且内容重复循环 D/Z6C&/I  
doStartTag, doAfterBody, doEndTag, release Z0x ar]4V  
@r/#-?W  
带内容的标签,且内容与JSP交互 ieDk;  
doStartTag, doEndTag, release, doInitBody, doAfterBody, release &S{RGXj_  
#%a;"w  
一个tag处理类可以通过javax.servlet.jsp.PageContext来与JSP交互,通过javax.servlet.jsp.PageContext类,tag处理类可以访问JSP中的request、session和application对像。 @Owb?(6?  
rdl;M>0@  
如果tag标签是互相嵌套的,那内层的tag处理类可以通过它的parent属性来访问上层的tag处理类。 7 E r23Q  
_E&A{HkJ  
一般情况都将所有的tag处理类打成了JAR的包,以便于发布。 Xb:;</  
.0S~872  
?*[N_'2W+  
Tag库描述(简称TLD) 3GaQk-  
3m]4=  
XX7{-Y y  
Tag库是用xml语言描述的,TLD包括了tag库中所有tag标签的描述,它一般用来被jsp服务器用来校验tag的语法正确性,或者被jsp开发者用来开发新的标签。 8 ##-EN;ag  
iEVb"w0 59  
TLD的文件扩展名必须为.tld,而且必须放在当前WEB应用的WEB-INF目录或其子目录中。 C]O(T2l{l  
rHC>z7+z.  
一个TLD的内容的开头必须遵守标准的XML开头,用于描述DTD和xml的版本,例如: 3 R=,1<  
UldXYtGe  
<?xml version="1.0" encoding="ISO-8859-1" ?> 9_q#W'/X  
j~2{lCT  
<!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN" "http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd"> *]ly0nP  
TLD必须以<taglib>来作为它的根元素,<taglib>的子元素如下表: IpINH3odT  
,~aQL  
_(_a*ml  
<taglib>的子元素 $*^kY;  
mC4zactv  
Element @oNH@a j%  
Description  ,V,`Jf  
#o=y?(  
tlib-version iQ tN Aj  
Tag库的版本 Vi>,kF.f V  
AYt%`Y.!  
jsp-version 0ZT5bg_M  
Tag库所需要的jsp的版本 `n#H5Oyn  
m;H.#^b*  
short-name t9,\Hdo  
助记符,tag的一个别名(可选) 8|):`u  
:Ux?,  
uri @G BxL*e  
用于确定一个唯一的tag库 =!kk|_0%E  
q 0$,*[PH  
display-name NO~*T?&  
被可视化工具(诸如Jbuilder)用来显示的名称(可选) v/Py"hQ  
[=E  
small-icon n(b(yXYm]  
被可视化工具(诸如Jbuilder)用来显示的小图标(可选) 6DM$g=/ '  
xAqb\|$^  
large-icon vL|SY_:4  
被可视化工具(诸如Jbuilder)用来显示的大图标(可选) V^7V[(~`  
xO$lsZPG  
description O ,J>/  
对tag库的描述(可选) 1 9&<|qTz  
gcE|#1>  
listener gi(H]|=a  
参见下面listener元素 ql<i]Y  
~?D4[D|sB  
tag =q>'19^Jx  
参见下面tag 元素 yL%K4$z  
*}]Nf  
Listener元素 @E^~$-J5j  
W 0(_ ~  
一个tag库可能定义一些类做为它的事件侦听类,这些类在TLD中被称为listener 元素,jsp服务器将会实例化这些侦听类,并且注册它们。Listener元素中有一个叫listener-class的子元素,这个元素的值必须是该侦听类的完整类名。 :?k>HQe  
2>|dF~"  
Tag元素 ]=.\-K  
g$7{-OpB  
每个tag元素在tag库中都要指出它的名字、类名、脚本变量、tag的属性。其中脚本变量的值可以直接在TLD中定义或通过tag附加信息的类来取得。每个属性描述了这个属性是否可以省略,它的值是否可以通过<%= …%>这样的JSP语法来获得,以及属性的类型。 ;,$NAejgd  
$$D}I*^Dt  
每一个tag在TLD中对应一个tag元素,下表是tag元素的子元素: us ;YV<)d  
[DJflCR&  
Tag元素的子元素 1q~+E\x  
JIOh#VNU  
元素名称 $"`- ^  
描述 N/]o4o  
b_,|>U  
name ~3UQ|j  
独一无二的元素名 _9faBrzd  
@En^wN  
tag-class ;lq;X{/  
Tag标签对应的tag处理类 - |kA)M[  
=CK4.   
tei-class -mC0+}h  
javax.servlet.jsp.tagext.TagExtraInfo的子类,用于表达脚本变量(可选) ?f#y1m  
9!f/aI  
body-content ~1cnE:x;V  
Tag标签body的类型 3Dg,GaRk  
v$~QU{ &  
display-name sqla}~CiX  
被可视化工具(诸如Jbuilder)用来显示的名称(可选) H70LhN  
,^?^ dB  
small-icon n/DP>U$I&  
被可视化工具(诸如Jbuilder)用来显示的小图标(可选) nS/)P4z  
0FGe=$vD  
large-icon uJOJ-5}yt  
被可视化工具(诸如Jbuilder)用来显示的大图标(可选) jH19k}D  
`o<' x.I  
description q:l>O5  
此tag标签的描述 r9X?PA0f  
Kpb#K[(]&  
variable c05TsMF&O  
提供脚本变量的信息(同tei-class)(可选) Z= jr-)kK  
cS>e?  
attribute z)'Mk[  
Tag标签的属性名 Rz (QC\(  
([ jF4/  
以下章节介绍对于不同类型的tag,如何具体地实现它们。 I'PeN0T f  
e&7JpT  
6Pnk5ps }h  
简单的tag =jSb'Vu|  
@b"t]#V(E  
e "/;7:J5\  
tag处理类 #~SP)Ukp  
dA@'b5N{"  
简单的tag处理类必须实现Tag接口的doStartTag和doEndTag方法。当jsp引擎碰到tag标签的开头时,doStartTag被调用,因为简单的tag没有body,所以此方法将返回 SKIP_BODY。当jsp引擎碰到tag标签的结尾时,doEndTag被调用,如果余下的页面还要被计算,那它将返回EVAL_PAGE,否则将会返回SKIP_PAGE。 r~N"ere26  
!xP8# |1  
以下是例子:对于标签 <tt:simple /> ,它的tag处理类实现如下: ZaeqOVp/j  
*&?c(JU;<  
Ae69>bkE0  
public SimpleTag extends TagSupport WtViW=j'  
j*F`"df  
{ 7w 37S  
eAX )^q  
public int doStartTag() throws JspException x\F,SEj  
kjEEuEv  
{ bA= |_Wt  
350_CN,  
try{ )_mr! z(S  
U"q/rcA  
pageContext.getOut().print(“Hello.”); Atflf2K  
8CnRi  
}catch(Exception e){ 8#(Q_  
@kk4]:,w  
throw new JspTagException(“SimpleTag: “ + e.getMessage()); ]04 e1F1J  
 yyv8gH  
} ,oaw0Vw  
d*A*y^OD  
return SKIP_BODY; Vnx,5E&  
_Dym{!t  
} Vy*:ne  
|AcRIq  
public int doEndTag() 'a$Gv&fu  
j6>.n49_  
{ ]Tkc-ez  
2kdC]|H2?  
return EVAL_PAGE; M&N B/  
)t$,e2FY  
} `D`sr[3n  
Mu{;vf|j  
} ,Y|^^?'j Q  
;3\3q1oX  
\2ZPj)&-E  
注意:如果tag标签没有内容的话,那必须定义body-content元素为空,例如 A-,up{g  
 emK$`9  
<body-content>empty</body-content> H|R T?Q  
{Zh>mHW3  
#ggf' QIHp  
带属性的tag标签
评价一下你浏览此帖子的感受

精彩

感动

搞笑

开心

愤怒

无聊

灌水
级别: 终身会员
发帖
3743
铜板
8
人品值
493
贡献值
9
交易币
0
好评度
3746
信誉值
0
金币
0
所在楼道
只看该作者 1 发表于: 2007-01-01
tag处理类 a@\D$#2r  
wk6NG/<  
对于tag标签的每个属性,你必须依照JavaBeans规范来定义其属性,以及get和set方法。以struts的logic:present 标签为例, -O&CI)`;B  
AC;V m: @{  
<logic:present parameter=”Clear”> Zs}5Smjl;%  
H ,+? t  
与此相应,此tag处理类应有如下方法和定义: &}?$i7x5  
!-N6l6N  
V5GW:QT  
protected String parameter = null; }~"hC3w  
={5#fgK>  
public String getParameter() "y_#7K  
VxY+h`4#  
{ {_Lg tu  
m;D- u>o  
return this.parameter; 6I!7c^]t  
>m# e:[N  
} K]j0_~3s  
LwhyE:1  
public void setParameter(String parameter) `2`\]X_A{  
nK$X[KrV'  
{ 7<jZ`qdq_  
zoDH` h_  
this.parameter = parameter; t=u  Qb=  
0H0-U'l  
} ?W0)nQU  
63.wL0~  
vjQb%/LWl  
注意:如果你的属性名为id,而且你的tag处理类是从TagSupport类继承的,那你就不需要定义它的属性和set和get方法,因为他们早已在TagSupport被定义过了。 V^hE}`>z&  
' j6gG  
Attribute元素 PH*\AZJCl  
cwm_nQKk  
对于tag标签的每个属性,你必须定义它是否必须的,它的值是否可以用诸如<%= …%>的表达式来获得,以及它的类型(可选),如果不指定它的类型,那就默认为是java.lang.String类型。如果rtexprvalue元素被定义为true或yes,那么在type元素中就定义了attribute的返回类型。 vk>b#%1{  
<`jLY)sw  
<attribute> = <j"M85.  
MB%Q WU  
<name>attr1</name> $8p7D?Y  
8a_[B~  
<required>true|false|yes|no</required> Q2A7mGN  
@ JvPx0  
<rtexprvalue>true|false|yes|no</rtexprvalue> &AlJ "N|  
?%VI{[y#>  
<type>attribute的返回类型(只用当rtexprvalue为真是才有效)</type> -F=v6N{  
$t5 V=}m>  
</attribute> ]$p{I)d&  
PG8^.)]M  
如果tag的某个属性不是必须的,那tag处理类会自动提供一个缺省值。 4;HJ;0-ps  
R(-<BtM!-  
例如,在logic:present这个tag标签中定义了一个属性叫parameter,但它不是必须的,而且它可以被诸如<%= …%>的表达式来赋值。 e5.h ?  
.<z!3O&L  
<tag> <BO)E(  
})w*m  
<name>present</name> sP-^~ pp  
\`ZW* EtPI  
<tag-class>org.apache.struts.taglib.logic.PresentTag</tag-class> 'kYwz;gp  
DAwqo.m  
<body-content>JSP</body-content> 4O:W#bx  
/B5-Fx7j3  
`)qVF,Z}  
vsL[*OeI  
<attribute> %L [&,a  
W5^.-B,(K  
<name>parameter</name> U:MPgtwe  
BC,.^"fA6  
<required>false</required>  be e5  
%+ FG,d  
<rtexprvalue>true</rtexprvalue> k<RZKwQc  
RNPbH.  
</attribute> cpt<WK}  
Z_gC&7+  
^1Xt]T`e  
k8,?hX:  
</tag> 341?0 %=  
}pa9%BQI  
^ FZ^6*  
属性元素的校验 .bVmqR`  
QRLJ_W^&u  
=&!HwOnp  
有关于tag标签的有效值可以从tag库的说明文档中获得,当JSP页面被编译时,jsp引擎会强制性地参照TLD中定义的规则进行检查。 F`nb21{0y&  
c9j*n;Q  
还有一个方法也可以进行属性元素的校验,就是先继承类TagExtraInfo,然后调用它的isValid方法。这个类同时也起到提供tag中定义的脚本变量信息的作用。 |0{u->+ )  
83cW=?UgA  
IsValid方法通过TagData对象来传递属性信息,它包括着tag的所有的属性名-值的信息。由于校验发生在运行时刻,因此这个属性的值将被赋值为TagData.REQUEST_TIME_VALUE。 aZKOY  
Z?J:$of*  
例如tag标签<tt:twa attr1=”value1” />在TLD中定义如下: NZN-^ >  
4f~["[*ea  
<attribute> .Hc(y7HV  
e4I^!5)N  
<name>attr1</name> &x-TW,#Ks  
M HL("v(@B  
<required>true</required> AM} brO  
:Vg}V"QR  
<rtexprvalue>true</rtexprvalue> ;eigOU]  
`]$?uQ  
</attribute> ;nmM7TZ;  
$jd<v1"o  
uhV0J97  
这个定义说明了attr1能在运行期间被赋值。 p.fF}B  
)4xu^=N&as  
以下的isValid方法检查attr1属性的值是否属于Boolean类型。注意由于attr1能在运行刻被赋值,那么isValid方法必须检查tag用户是否对此tag进行了运行时刻赋值。 AGYm';z3  
m}$+Hdk+7  
2JmZ{  
Public class TwaTEI extends TagExtraInfo U-0#0}_  
X E}H3/2  
{ "0jJh^vk  
iA!7E;o  
public boolean isValid(Tagdata data) -~)OF  
^IgY d*5  
{ %;z((3F  
,J mbqOV?!  
Object o = data.getAttribute(“attr1”); ZX b}91rzt  
{OT:3SS7  
If(o != null && o != TagData.REQUEST_TIME_VALUE) "`A:(<x  
LF*Q!  
{ r)1Z(tl  
dMl+ko  
if( ( (String)o).toLowerCase().equals(“true”) || %K 4  
7s'- +~  
((String)o).toLowerCase().equals(“false”) ) -%IcYzyA  
tn/T6C^)  
return true; TTI81:fku  
tUE'K.-  
else daZQz"PP  
~3WL)%  
return false; h`MdKX$  
OQlG+|  
} Z& !!]"I  
3$M3Q]z  
else m.%`4L^`T  
us)*2`?6t  
return true; ^9"KTZc-*  
g?/XZ5$a5  
} 83 ]PA<R  
=N.!k Vkl  
} I#A2)V0P)  
,ul5,ygA  
o96:4j4  
带body的tag 8IY19>4'5J  
es.CLkuD7Y  
0.(<'!"y  
tag处理类 4\ $3  
*07sK1wW  
Yx?aC!5M  
如果tag标签含有内容,那处理方式会略微有些不同,而且还要视tag处理类是否要与body交互的情况而定。如果要与body交互,那我们认为tag处理类要可能要对body进行操作。 Wr|G:(kw\!  
@YL}km&Fw  
Tag处理类不与body交互 A! 6r/   
)Id2GV~2B  
如果tag处理类不与body交互,tag处理类应该实现Tag接口或从TagSupport中派生,如果body要被计算,那么doStartTag方法应返回 EVAL_BODY_INCLUDE,否则应返回SKIP_BODY。 H30OUrD  
#n})X,ip2  
如果tag处理类要对body反复运算,则它应该实现IterationTag或从TagSupport中派生。如果tag处理类认为body还未计算完的话,那它的doStartTag方法和doAfterBody方法必须返回EVAL_BODY_AGAIN。 $Tl<V/  
P2k7M(I_&  
Tag处理类与body交互 k 'zat3#f  
Ky9W/dCR  
如果tag处理类与body交互,那tag处理类应实现BodyTag接口或从BodyTagSupport中派生。这种tag处理类一般要实现doInitBody和doAfterBody方法。 Q</h-skLZ  
Pern*x9$  
Body允许一些方法来读写它的内容。Tag处理类可以调用body内容的getString或getReader方法来从body中提取信息,也可用 writeOut(out) 方法来将body的内容写入到out对象中。其中out对象通过tag处理类的getPreviousOut方法来获得。 y$oW!  
cx(2jk}6  
如果body的内容需要被计算,那么doStartTag方法必须返回EVAL_BODY_BUFFERED,否则,它将返回 SKIP_BODY。 -}<g-*m"q  
VuWib+fT  
doInitBody 方法 F1u)i  
E/ O5e(h  
此方法在body内容已经设好,但未被计算之前被调用。你可以根据不同的body内容来制定初始化策略。 3IyZunFT  
RV!<?[  
doAfterBody方法 .9 kyrlm  
9m!7|(QV  
此方法在body内容已被计算后进行调用。 <'*4j\*  
z}APR@?`n8  
和doStartTag方法一样,doAfterBody方法返回一个指示符指示是否要继续计算body,如果要继续计算,则doAfterBody应返回EVAL_BODY_BUFFERED,否则,它应返回SKIP_BODY。 fi=?n{e'  
1X@b?6  
release 方法 GXeAe}T  
!C`20,U  
tag处理类调用此方法将它的状态重置为初始状态,并释放所有的私有资源。 rWr/p^~  
J:};n@<  
YX;nMyD?~  
以下的例子读取body的内容(其中含有一条sql语句),然后将它传递给一个对象,让它进行查询。由于此处body不须重新计算,所以doAfterBody会返回SKIP_BODY。 =~,l4g\  
T&/_e   
QMb^&?;s  
Public class QueryTag extends BodyTagSupport 5P\N"Yjx'  
oyQ0V94j  
{ #: hVF/  
U"x~Jb3]O  
public int doAfterBody() throws JspTagException oFyeH )!  
,>S+-L8  
{ tR O IBq|  
ADBpX>  
BodyContent bc = getBodyContent(); E\! n49  
vf_OQ4'G,  
//将body的内容以字符串的格式提取出来 hho%~^bn(  
\^s2W:c  
String query = bc.getString(); 0BXs&i-TP5  
QK7e|M  
//清除body '>]&rb09|  
?Ccw4]YO,=  
bc.clearBody(); xppkLoPK  
f !!P  
try{ J!<#Nc  
^Y<M~K972  
Statement stmt = connection.createStatement(); x\]z j!  
$kv[iI @  
Result result = stmt.executeQuery(query); Z"N(=B  
eaxfn]gV  
}catch(SQLException e){ F,.Q|.nN  
1gk0l'.z  
throw new JspTagException(“queryTag: “ + e.getMessage() ); ex0oAt^  
u#!GMZJN  
return SKIP_BODY; c.-cpFk^L&  
O(Td:Zdp  
} p_gA/. v=  
_f66>a<  
} d}VALjXHX!  
A`Q >h{  
6Zw$F3 <  
body-content元素 W%ix|R^2]  
Ja*,ht(5  
E{JTy{z-  
由于tag可能会有body,你必须用body-content元素来指定body内容的类型: tQ; Fgv8Y!  
i=b'_SZ '  
<body-content>JSP|tagdependent</body-content> \)\n5F:Zu  
._x"b5C  
Bq~S=bAB>R  
如果body的内容是定制的或内部的tag、脚本元素、或HTML广本,则归类为JSP类型。其他的类型,比如上面代码所述的?D?D将sql statement类传给 query tag的这种类型应该标为tagdependent。 5?>Q[a.Ne  
ZU B]qzmK  
注意:实际上body-content的值并不影响tag处理类对body内容的处理,它仅仅是被tag编辑工具用来描述此body的内容。 w\d1  
md{1Jn"  
7]L}~  
用tags定义脚本变量 l@UF-n~[  
-6F\=  
J^` pE^S  
tag处理类 DPnrzV )  
}o GMF~  
NODE`VFu  
tag处理类负责创建或设置页面中定义的脚本变量,用pageContext.setAttribute(name,value,scope)或pageContext.setAttribute(name,value)方法来实现。一般来说,tag处理类通过脚本变量的名称来获取它,脚本变量的名称一般可用get方法来获得。 T^|6{ S\  
^|rzqXW  
如果脚本变量的值依赖于tag处理类中的上下文中某一对象,那它可用pageContext.getAttribute(name,scope)方法来找到那个对象。一般的处理过程是tag处理类先找到脚本变量,再对其进行处理,然后用pageContext.setAttribute(name,object)的方法来设置它的新值。 0""%@X]m  
6S%KUFB+e  
对象的生存周期(scope)如下表: p0Ij 4   
= "Lb5!  
对象的生存周期表 :8)3t! A  
Vz,"vBds  
名字 "$%&C%t  
可访问范围 G7-.d/8|^  
生存周期 )J\ JAUj  
yq[CA`zVN  
page S|RUc}(  
当前页面 P)=$0kR3  
一直有效,除非页面向客户提交响应或重定向到一个新页面 0[Zs8oRiI  
MGKSaP;x  
request ^?0WE   
当前页面或当前页面重定向到的页面 z*^vdi0  
一直有效,除非页面向客户提交响应 v>Kv!OY:c  
4NFvX4  
session F+Hmp\rM#  
当前页面或在同一浏览器窗口中的页面 b Oh[(O!  
一直有效,除非关闭当前浏览器、超时、网络故障 jA%R8hdr_  
'! ;Xxe5  
application $kQQdF  
整个web应用程序的所有请求 vN)l3  
一直有效,除非发生网络故障、服务器故障 QbAEW m  
60 p*4>^v  
提供关于脚本变量的信息 eNt1P`2[  
W2,Uw1\:1  
以下示例定义了一个名为“book”的脚本变量,用来访问程序中关于书的信息: QfU{W@!h  
c$%I^f}'  
<bean:define id=”book” name=”bookDB” property=”bookDetails” type=”database.BookDetails” /> Fmrl*tr  
+-:G+9L@  
<font color=”red” size=”+2” > pe`&zI_`?  
iJ`v3PP  
<%= messages.getString(“CartRemoved”) %> )d?L*X~y'  
,?!4P+ob  
<strong><jsp:getProperty name=”book” property=”title” /></strong> /kd6Yq(y  
?m.Ry  
</font> $k3l[@;hE  
EtvYIfemr  
当包含此tag的JSP页面被编译时,jsp引擎会自动生成关于此book的同步的代码(同步可以避免几个客户同时访问此book时造成的冲突),要生成同步代码,jsp引擎需要知道此脚本变量的如下信息: |f&=9%  
bU"2D.k  
· 脚本变量名称 KA0_uty/T  
2Yd;#i)  
· 脚本变量所属的类 >9]i#So^  
4^BHJOvs  
· 此脚本变量是否引用了一个新的或已存在的对象 +D+Rf,D  
SE!0f&  
· 此脚本变量的有效性 .3 T#:Hl  
m3K .\3  
有两种办法可以向jsp引擎提供关于脚本变量的信息:在TLD中定义variable子元素,或用tei-class子元素定义一个额外tag信息类。用variable最简单,但可能降低了一些灵活性。 =arrp:  
ob.=QQQs  
Variable元素 &!FWo@  
m:XMF)tW  
Variable元素有如下子元素: -}*YfwK  
a)Ca:p  
· name-given ?D?D 给出的名字,是一个常量 q--;5"=S  
dD8f`*"*=  
· name-from-attribute?D?D 属性名,在编译时给出的属性名 wi/dR}*A  
@5\ns-%  
name-given或name-from-attribute两者必须选一,但以下子元素是可选的: VpkkiN  
<O?UC/$)7  
· variable-class?D?D变量的类型,缺省为java.lang.String。 4!KUPgg  
[6Gb@jG  
· declare?D?D此脚本变量是否引用了一个新对象,缺省为True。 u CNi&.  
$]yHk  
· scope?D?D脚本变量的范围,缺省为NESTED。下表描述了scope的几种类型: o=VDO,eS  
${F] N }  
脚本变量的有效范围 \.<V~d?  
Lk|%2XGO&  
?N*|S)BN  
有效性 k9<P]%  
方法 tk <R|i  
7 a}qnk %  
NESTED yqw#= fy  
在tag标签的开始和结束之间 \L: ;~L/  
如果tag处理类实现BodyTag接口,则在doInitBody和doAfterBody中调用,否则在doStartTag中调用 L(2KC>GvA  
le-Q&*  
AT_BEGIN n^ AQ!wC  
从tag标签的开始一直到页面结束 80M"`6  
如果tag处理类实现BodyTag接口,则在doInitBody、doAfterBody和doEndTag中调用,否则在doStartTag和doEndTag中调用 -C|1O%.  
X_eh+>D  
AT_END vA*Ud;%R  
从tag标签的结束一直到页面结束 75LIQ!G|=  
在doEndTag中调用 Je#vl4<L  
26,!HmtC  
以struts为例,它的bean:define标签的实现遵循JSP1.1规范,此规范要求使用额外tag信息类来定义脚本变量。Variable元素是JSP1.2规范中加入的。以bean:define标签为例,你可以定义如下variable元素: (hn@+hc  
D^knN-nZ*  
<tag> [y}0X^9,E  
HNRZ59Yyq  
<variable> -&u2C}4s  
yqm^4)Dp  
<name-from-attribute>id</name-from-attribute> 8)D5loS  
.*N,x0 B(  
<variable-class>database.BookDetails</variable-class> <4{,u1!t  
m\88Etl@  
<declare>true</declare> _^<HlfOK  
Ls#= R  
<scope>AT_BEGIN</scope> <= xmJx-V  
h98_6Dw(]  
</variable> s^]F4'  
S&A, Q'  
</tag> X/_e#H0  
Jbud_.h9  
额外tag信息类 ,{ C   
DA_[pR  
如果要定义一个额外tag信息类,你要继承javax.servlet.jsp.TagExtraInfo类。一个TagExtraInfo类必须实现getVariableInfo方法,此方法返回一个叫VariableInfo的数组类,它包括如下信息: t UJ m}+=>  
,iXE3TN;W  
· 变量名 ]E1aIt  
Rl7V~dUY  
· 变量所属类名 >g!a\=-[  
]OE{qXr{  
· 此变量是否引用了一个新对象 =gCv`SFW  
7.n/W|\  
· 此变量的有效范围 li4rK <O  
*<yKT$(+_  
jsp引擎将一个名为data的参数传给getVariableInfo方法,data中包括tag标签中的所有“属性名?D?D属性值”对。它可以用来向VariableInfo对象提供脚本变量的名字和类名。 D 6trqB  
`zNvZm-E  
以struts为例,它在bean:define标签中定义了一个名为DefineTei的额外tag信息类,用来向脚本变量提供信息。由于脚本变量的名称(book)和类名(database.BookDetails)是通过tag标签的属性来传递的,它们一般定义在VariableInfo的构建代码中,并且可用data.getAttributeString方法来得到这些信息。如果要允许book脚本变量能在从tag开始直到整个JSP页面结束的范围内都可用的话,那它的范围应设为AT_BEGIN。如下所示: -'FzH?q:  
0f).F  
2jTP (b2b  
public class DefineTei extends TagExtraInfo bJJB*$jW=  
+1j@n.)ft  
{ N 4$!V}pp  
`B+P$K<X  
public VariableInfo[] getVariableInfo(TagData data) VjbRjn5LI  
gdupG  
{ %(GWR@mfC  
*]Nd I  
String type = data.getAttributeString(“type”);  )_P|_(  
HhN;&67~Z  
If( type == null) | I_,;c  
=.9L/74@  
type = “java.lang.Object”; `+[e]dH  
Y/1KvF4)k  
return new VariableInfo[] { Y27x;U  
xEN""*Q  
new VariableInfo(data.getAttributeString(“id”), ^EGe%Fq*x]  
zY\pZG  
type, HS"E3s8  
[:-Ltfr  
true, tG(#&54  
29DYL  
VariableInfo.AT_BEGIN) 8A{n9>jrb  
A;nrr1-0  
}; 8<UD#i@:C  
*NdSL  
} .4c*  _$  
Tbl~6P  
} uGIA4CUm  
7iCH$}  
lP*_dt9  
注意:关于额外tag信息类的类名必须要在TLD中的tag标签下的tei-class子元素中定义。因此,DefineTei的tei-class中的定义看起来如下: B#RwW,  
<tei-class> bD_|n!3  
org.apache.struts.taglib.bean.DefineTagTei { -*+G]  
</tei-class> SoZ$1$o2  
t sdkpt  
_-aQ.p ?T  
具有协作关系的tag BdcTKC  
QXZjsa_|  
?N2/;u>  
tag通过共享对象来进行协作,JSP技术支持2种方式的对象共享。 4fPbwiK j  
*-~B{2b<  
第一种方法是使用pageContext对象进行对象的共享(可支持JSP页面和tag处理类之间的共享),如果在一个tag处理类中要调用由另一个tag处理类创建的对象,可调用pageContext.getAttribute(name, scope)方法。 9%e& Z'l  
f/t1@d!  
第二各方式的共享是对于tag之间有嵌套关系而言的,外层的tag所创建的对象对于内层的tag来说是可以共用的。这种形式的共享的好处是减少了可能存在的重名冲突。 40}qf}8n t  
lhsd 39NM  
要访问一个嵌套tag创建的对象,tag处理类必须先找到此嵌套tag对象,可用TagSupport的静态方法 TagSupport.findAncestorWithClass(from, class)或TagSupport.getParent方法。前者在当不确定此tag是否为嵌套tag对象时使用。一旦它的父类被找到,它就能访问其所有动态或静态创建的对象。静态创建的对象是父类的成员,而动态创建的对象可能是父类的私有对象。诸如此类的对象可以用tag处理类的setValue方法来保存,用getValue方法来获得。 Q2sX7 cE  
t_HS0rxG  
下例阐述了以上两种共享对象的方法。在这个例子当中,一个查询tag检查一个名为connection的属性名是否在doStartTag中被设置。如果connection属性被设置,tag处理类从pageContext中得到这个connection对象。否则,此tag处理类先找到它的父tag处理类,然后从它的父tag处理类中找到connection对象。 ~^*IP1.3  
i$HA@S  
VT1Nd  
public class QueryTag extends BodyTagSupport 1:]iV}OFqR  
B. Rc s  
{ OYKeu(=L  
K7 >Z)21  
private String connectionId; dn0?#=  
'f0*~Wq|  
public int doStartTag() throws JspException }HB)%C50.  
5~Vra@iab:  
{ 8&FnXhZg4  
k,o|"9H  
String cid = getConnection(); b|F_]i T  
?n o.hf  
if(cid != null) b#%s!  
,c#IxB/0  
{ k2Yh?OH  
^c?2n  
//存在一个connection id,使用它。 )rJ{}U:S  
=Q985)Y&  
connection = (Connection) pageContext.getAttribute(cid); W =YFe<Q  
?c7 12a ?  
} =!{7ZSu\  
]k~k6#),;  
else rj zRZ  
J@RhbsZn  
{ _[|~(lDJl  
k.nq,  
ConnectionTag ancestorTag = (ConnectionTag)findAncestorWithClass(this, 7{HJjH!zx  
P,y*H_@k  
ConnectionTag.class); ceN*wkGyB  
}C1&}hZ  
if(ancestorTag == null) (`+%K_  
JR/:XYS+  
{ ((i%h^tGa;  
(. ~#bl  
throw new JspTagException(“一个没有connection属性的查询标签必须被一个connection标记嵌套。”); gMe)\5`\Y  
ta`}}I  
} G+Ft2/+\  
` #Qlr+X  
connection = ancestorTag.getConnection(); df_hmkyj  
s=0z%~H  
} g?,\bmHE  
k<p$BZ  
} hl`4_`3y  
v'`VyXetl  
} 9PXG*r|D  
y8"8QH  
b77Iw%x7  
c#'t][Ii  
此查询标签在JSP页面中的调用形式可以从以下2种定义中任选一种: Rlc$2y@pU  
\"!Fw)wj  
)}QtK+Rq  
<tt:connection id=”con01” ...> ... </tt:connection> &%M!!28X:  
P|,@En 1!  
<tt:query id=”balances” connection=”con01” > = GUgb2TAT  
3^m0 k E  
SELECT account, balance FROM acct_table ph*?y  
UIS\t^pJD  
where customer_num = <%= request.getCustno() %> f' |JLhs  
pVgzUu7  
</tt:query> ] %A mX-U  
3[mVPV  
, y%!s27  
G 3))3]  
HCOv<k  
<tt:connection ...> J~.8.]gXW  
1F`1(MYt9  
<x:query id=”balances”> VX+:k.}  
FbH 1yz  
SELECT account, balance FROM acct_table \0gU)tVZ  
vjNP  
where customer_num = <%= request.getCustno() %> )fPN6x/e  
)1#J4  
</x:query> fpd4 v|(  
Mn`);[  
</tt:connection> Jz"Yb  
~kHWh8\b:  
g!/O)X3  
与此同时,在TLD中必须指定connection属性为可选的,定义如下: -M\ae  
y Skz5K+|g  
<tag> W(~7e?fO  
.W;,~.l  
... ,O`a_b]  
{tE9m@[AF  
<attribute> ~w Ekbq=  
pi?$h"y7Q  
<name>connection</name> L_@P fI  
zDakl*  
<required>false</required> LN=6u  
<c; U 0! m  
</attribute> ' ozu4y  
TR7j`?  
</tag>
描述
快速回复

您目前还是游客,请 登录注册
如果您在写长篇帖子又不马上发表,建议存为草稿
认证码:
验证问题:
10+5=?,请输入中文答案:十五