标准的JSP 标记可以调用JavaBeans组件或者执行客户的请求,这大大降低了JSP开发的复杂度和维护量。JSP技术也允许你自定义taglib,其实换句话说,taglib可以看成是对JSP标记的一种扩展,正如xml是对html的一种扩展一样。taglib通常定义在tag标签库中,这种标签库存放着你自己定义的tag标签。简而言之,如果使用taglib,那么你可以设计自己的JSP标记!
me1ac\ 7xwS
.| 一般来说,自定义tag标签主要用于操作隐藏对象、处理html提交表单、访问数据库或其它企业级的服务,诸如邮件和目录操作等等。自定义tag标签的使用者一般都是那些对java编程语言非常精通,而且对数据访问和企业级服务访问都非常熟悉的程序员,对于HTML设计者来说,使得他可以不去关注那些较复杂的商业逻辑,而将精力放在网页设计上。同时,它也将库开发者和库使用者进行合理分工,自定义tag标签将那些重复工作进行封装,从而大大提高了生产力,而且可以使得tag库可用于不同的项目中,完美地体现了软件复用的思想。
'}P)iS2 <H}"xp)j0 在这篇文章中,我们主要讨论:
EK@yzJ% KP_=#KD · 什么是自定义tag标签?
H#m)`=nZSZ 7Q0M3m · 怎么使用tag标签?
Q7"KgqpQ3 ~bigaY o 声明要使用的tag库
.oaW#f}0P miZ{V% o 找到与之对应的tag处理类
A.
U< @`wBe#+\ o tag标签的类型
q jDWA' (66X · 自定义tag标签
gLl?e8[F pF K[b o tag处理类
z+PSx'#} _f|Au`7m o tag库描述
DcSL f4A C(?>l.QGw o tag标签示例
;)0vxcMB kQ.atr`? e o 带属性的tag
EVgn^, T"kaOy o 带body的tag
mRj-$:}L rU<
H7U o 定义了脚本变量的tag
x:xKlPGd Ad@))o2 o 具有协作关系的tag
F8_pwJUpf- P%'bSx1 · 自定义tag标签
"!E(=W? n_$lRX5 o 一个迭代tag的例子
?tqTG2! ( e>nRJH8pK o 一个模板tag库
,EcmMI^A DG7FG-- o tag处理类到底是怎样被调用的?
(z ;=3S <g>_#fz"K 2?QIK3"v 什么是自定义的tag?
#Sb1oLC v}xz`]MW<, 一个自定义的tag标签是用户定义的一种JSP标记。当一个含有自定义的tag标签的JSP页面被jsp引擎编译成servlet时,tag标签被转化成了对一个称为tag处理类的对象进行的操作。于是当JSP页面被jsp引擎转化为servlet后,实际上tag标签被转化成为了对tag处理类的操作。
AJt0l|F y"e'Gg2 自定义tag标签有很多特色,诸如:
wMt?yc:X Y)c9]1qly · 可以在JSP页面中自定义tag标签的属性
X]C-y,r[M kul&m| · 访问JSP页面中的所有对象
~;UK/OZ )uwpeq$j7l · 可以动态地修改页面输出
\{{B57/Isq zJ|Ek"R. · 彼此这间可以相互通信。你可以先创建一个JavaBeans组件,然后在一个tag中调用此JavaBeans组件,同时可以在另一个tag中调用它。
Yi7`iC b'Mg · tag允许相互嵌套,可以在一个JSP页面中完成一些复杂的交互。
d";+8S cFGP3Q4{ !uO|1b 使用tag标签
Ywr^uy1V,/ t.lm`= A[htG\A` 0 本节主要描述怎样在JSP页面中使用tag标签,以及tag标签的不同类型。
l=
~]MSwY >W.Pg`'D 要使用tag标签,JSP程序员必须做2件事:
B964#4&
9 >I]t|RT]) · 声明此tag标签的tag库
TL]2{rf~ >/1.VT\E · 实现此tag标签
"JJ )w0 aODOc J N 声明tag标签所在的tag库
|;OM,U2 ZN%$k-2 如果要使用tag标签,则应用JSP的taglib指示符来指定其tag库(注意:taglib要在在使用此tag标签之前声明)
?Q9/C| :'1ePq <%@ taglib uri=”/WEB-INF/tutorial-template.tld” prefix=”tt” %>
hJhdHy=U FK@rZP uri属性定义了唯一的标签库描述(以下简称TLD),它可以是直接是tld文件名或一个独一无二的名字。prefix是用来区别其它TLD中和本TLD中有重名的tag的一种手段。
j\@s pbE@ iknB c-TLD TLD必须以.tld作为扩展名,并且存放在当前应用的WEB-INF目录或其子目录下。你可以通过它的文件名直接引用它,也可以通过别的方式间接地引用它。
)3h=V^rm Q&`$:h.~ 以下taglib指示符直接引用一个TLD:
qIA!m
.GC f
IQ$a> <%@ taglib uri=”/WEB-INF/tutorial-template.tld” prefix=”tt” %>
!?O:%QG z[ z'.{;D 以下的taglib指示符通过一个逻辑名称间接地引用一个TLD:
p*#SSR9< [7|}h/ <%@ taglib uri=”/tutorial-template” prefix=”tt” %>
;op+~@*! qO&:J\d 如果是间接引用TLD的话,那你必须还要在web.xml中定义此逻辑名称与tld文件之间的映射,具体做法是在web.xml中加入一个名为taglib的元素:
e3)rF5pp F~W*"i+EZ <taglib>
,dzbI{@6 78dmXOZ'_h <taglib-uri>/tutorial-template</taglib-uri>
.Pxb9mW kRSu6r9 <taglib-location>
'PV,c|f> JS({au /WEB-INF/tutorial-template.tld
WQiEQ>6(t( .LnXKRd{ </taglib-location>
v SHb\V# &Vnet7LfU </taglib>
@iC!Q>D J>!p^|S{ I4qzdD 实现此tag标签
\Qu~iB(Y VI" ,E} =2J+}ac 为了实现tag标签,你有2种方法来存放tag处理类。一、让tag处理类以.class的方式存放于当前应用的WEB-INF/class子目录下,二、如果tag处理类是以JAR包的形式存在的话,那可以放在当前应用的WEB-INF/lib目录下,如果tag处理类要在多个应用中共享,那么它就应放在jsp服务器上的common/lib目录下,对于tomcat来说,就是tomcat/common/lib目录下。
,0lRs sGMC$%e} [gIStKe tag标签类型
|I)xK@7 iu*u|e h-lMrI)U?h 自定义的tag标签遵循XML语法。它有一个开始标记和一个结束标记,有的还有body(即文本节点):
+;FF0_ "Q2[A]4E <tt:tag>
6$fC
R cl:*Q{(Cjk body
AGK+~EjL@ g@B9i= </tt:tag>
#\%GrtM t~sW]<qjp MT%ky 一个不带body的tag标签如下:
s![=F}ck 5A~w_p*} <tt:tag />
3w!oJB 1hi^ \&ERSk2 简单的tag标签
GlQ=M )E (t<i?>p 一个没有body和属性的tag标签如下:
g>OGh o k?|VFh1 <tt:simple />
ScZ$&n N;r,B ;u}MG3Y8 带属性的tag标签
oJyC{G X=${`n%LG c7wza/r> 自定义标签可以有自己的属性。属性一般在开始标记中定义,语法为 attr=”value”。属性的作用相当于自定义标签的一个参数,它影响着tag处理类的行为。你可以在TLD中详细定义它。
`1M_rG1/+ PM%./ 你可以用一个String常量给一个属性赋值,也可以通过表达式给它赋值,如<%= ...%>。以struts为例,它的logic:present标签就是用的String常量来给属性赋值:
P4R.~J ;8 Qbt
fKn95 <loglic:present parameter = “Clear”>
|])%yRAGQ ,1^)JshZ~ 而另一个标签logic:iterate是用表达式来给属性赋值:
zs[t<`2 //H+S
q66 <logci:iterate collection=”<%= bookDB.getBooks() %>”
_or$^.=' X903;&Cim id=”book” type=”database.BookDetails”>
_I5p
7X '
nf"u >a_K:O|AJ 带body的tag标签
1;ZEuO ?em )om 一个自定义标签可以包含其它自定义标签、脚本变量、HTML标记或其它内容。
<KHB/7 O}IS{/^7 在下述例子中,此JSP页面使用了struts的logic:present标签,如果些标签定义了parameter=”Clear”的属性,则将清除购物车的内容,然后打印出一条信息:
bsqoR8 Q6Jb]>g\H <logic:present parameter=”Clear”>
G!0|ocE} O}#*U+j <% cart.clear(); %>
M 80U s. Pvbw>k; <font color=”#ff0000” size=”+2”><strong>
b*P\a HV>|f'45 你选择了清除购物车!
ks(PH6:]< "UVV/&`o </strong></font>
#cjB <APY ^9cqT2:t </logic:present>
{Z-5
JhB{aW> M&Ycw XV:Z 到底是用属性还是用body来传递信息?
q' _ :V+t|@m5l 如上所述,我们既可以通过属性,也可以通过body来传递信息。但一般来说,比较简单的类型,如字符串或简单表达式最好采用属性来传递信息。
Eptsxyz{ Kq-y1h]7H aASnk2DFd 定义脚本变量的tag标签
pC#Z]_k v,g,c`BjK 所谓脚本变量,是指JSP中可以调用的变量或对象。它可由tag标签产生。以下示例阐述了一个tag标签定义了一个名为tx的由JNDI所定义的事务处理对象。脚本变量可以是ejb对象、事务、数据库连接等等:
3b%y+?-{\u W=F?+KgL <tt:lookup id=”tx” type=”UserTransaction” name=”java:comp/UserTransaction” />
[0)iY%^ eYsO%y\I <% tx.begin(); %>
W{Nhh3 '-W
p|A ...
Y;-" Z zg8m(=k' IXd&$h]Lq 具有协作关系的tag标签
~j F5%Gu r"5]U`+ 自定义tag标签之间可以通过共享对象来实现协作。在下述例子中,标签tag1创建了一个名为obj1的对象,在标签tag2仍可以重复使用obj。
$2;YJjz(
n-H0cm <tt:tag1 attr1=”obj1” value1=”value” />
H3`%#wQ0j L6l~!bEc <tt:tag2 attr1=”obj1” />
m#%5H ]!0*k#i_. 在以下这个例子当中,如果外层的tag标签创建了一个对象,那么其内层的所有tag标签都可以使用这个对象。由于这样产生的对象没有一个指定的名字,那么就可以将少重名的冲突。这个例子阐述了一系列协作的嵌套对象。
=_
-@1
1a 5%tIAbGW <tt:outerTag>
nwO;>Qr
ckhW?T>l <tt:innerTag />
tk1qgjE(? +twBFhS7k </tt:outerTag>
?+`Zef.g <
> f12pu hr]NW>; Tag处理类
1iF
|t5>e WGp81DNS| 0m*0I> Tag处理类必须实现Tag接口或BodyTag接口,不过现在一般都流行从TagSupport或BodyTagSupport类中继承,这些类或接口都可以在javax.servlet.jsp.tagext包中找到。
*pI3"_ 2"V?+Hhz 当JSP引擎看到自己的JSP页面中包含有tag标签时,它会调用doStartTag方法来处理tag标签的开头,调用doEndTag方法来处理tag标签的结束。
#c?\(qjWA tw*qlb FHv 下表说明不同类型的tag所需要不同的处理过程:
)O2^?Q quS _NqEhf:8 Tag处理类的方法
"%>/rh2Iq (VBoZP=W Tag标签类型
3fM 所调用的方法
HC!$Z`}Y RJBNY;0 基本标签
C(W?)6? doStartTag, doEndTag, release
IybMO5Mwn yKfRwO[j 带属性的标签
;=UrIA@y;= doStartTag, doEndTag, set/getAttribute1...N, release
W P.6ea7k 4(B,aU>y 带内容的标签
zFQxW4G doStartTag, doEndTag, release
6PJ0iten Fnll&TF 带内容的标签,且内容重复循环
|q5\1}@: doStartTag, doAfterBody, doEndTag, release
??1V__w aEX+M57k~ 带内容的标签,且内容与JSP交互
=>ph\ doStartTag, doEndTag, release, doInitBody, doAfterBody, release
-Frx {3 G]q6Ika 一个tag处理类可以通过javax.servlet.jsp.PageContext来与JSP交互,通过javax.servlet.jsp.PageContext类,tag处理类可以访问JSP中的request、session和application对像。
~>#=$#V :Q&8DC#] 如果tag标签是互相嵌套的,那内层的tag处理类可以通过它的parent属性来访问上层的tag处理类。
S@4p.NMU HD%n'@E 一般情况都将所有的tag处理类打成了JAR的包,以便于发布。
Y(mnGaVn %/; *Ewwb W?TvdeBx Tag库描述(简称TLD)
-<0xS.^ 8SGqDaRt -8zdkm8k Tag库是用xml语言描述的,TLD包括了tag库中所有tag标签的描述,它一般用来被jsp服务器用来校验tag的语法正确性,或者被jsp开发者用来开发新的标签。
9u?[{h.`B |-l9 Z TLD的文件扩展名必须为.tld,而且必须放在当前WEB应用的WEB-INF目录或其子目录中。
nz2`YyR FHbyL\Q 一个TLD的内容的开头必须遵守标准的XML开头,用于描述DTD和xml的版本,例如:
>bRoQ8 0q6xXNAX <?xml version="1.0" encoding="ISO-8859-1" ?>
*(g0{V h[ cqa <!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN" "
http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd">
yOTC>?p% TLD必须以<taglib>来作为它的根元素,<taglib>的子元素如下表:
W*;r}!ro *q-VY[2 dkWV/DAm <taglib>的子元素
0<FT=tKm
i"`N5 Element
zFi)R }Ot Description
w<LV5w+ 8hV>Q tlib-version
LK|1[y^h Tag库的版本
XOL_vS24 kxCN0e#_ jsp-version
`T,^os#6 Tag库所需要的jsp的版本
]bP1gV(b- oM^VtH=> short-name
@rP#ktz] 助记符,tag的一个别名(可选)
lJ
Jn@A |mQC-=6t;Y uri
ntxaFVD 用于确定一个唯一的tag库
50e
vWD ku)/
8Z`$ display-name
7Y.mp9, 被可视化工具(诸如Jbuilder)用来显示的名称(可选)
yZ-Ql11 BU<Qp$& small-icon
z2iWr 被可视化工具(诸如Jbuilder)用来显示的小图标(可选)
5cj&D74o 8I~*9MUp large-icon
Ca["tks 被可视化工具(诸如Jbuilder)用来显示的大图标(可选)
4axuE] Tmk'rOg5 description
*|OUd7P:hU 对tag库的描述(可选)
T"DG$R,Aj e7yn"kd listener
siOyp] 参见下面listener元素
W'PW;., P N(<=v&E tag
FI @kE19 参见下面tag 元素
W[LQ$uj 8&:dzS Listener元素
.[
s6x5M &8dj*!4H 一个tag库可能定义一些类做为它的事件侦听类,这些类在TLD中被称为listener 元素,jsp服务器将会实例化这些侦听类,并且注册它们。Listener元素中有一个叫listener-class的子元素,这个元素的值必须是该侦听类的完整类名。
TU1W!=Z F8J\#PW Tag元素
QsPZ dC f3*SIKi 每个tag元素在tag库中都要指出它的名字、类名、脚本变量、tag的属性。其中脚本变量的值可以直接在TLD中定义或通过tag附加信息的类来取得。每个属性描述了这个属性是否可以省略,它的值是否可以通过<%= …%>这样的JSP语法来获得,以及属性的类型。
WM0-F@_ WeIi{<u8R 每一个tag在TLD中对应一个tag元素,下表是tag元素的子元素:
bSW~hyI w Ow{NI-^K Tag元素的子元素
|ADg#oX xf;Tk 元素名称
CN6b982& 描述
:n OCs v<ati c name
M1eM^m8U 独一无二的元素名
8#|PJc $*j)ey> tag-class
~PX#' Jr Tag标签对应的tag处理类
{3BWT dW%;Z tei-class
hM[I}$M&O javax.servlet.jsp.tagext.TagExtraInfo的子类,用于表达脚本变量(可选)
?:7.3{|Aq Me-H'Mp~ body-content
#U6~U6@ Tag标签body的类型
} DjbVYH 7G%^8
ce{! display-name
qJK6S4O] 被可视化工具(诸如Jbuilder)用来显示的名称(可选)
Gdr7d `jT1R!$3F small-icon
/<\do 1 被可视化工具(诸如Jbuilder)用来显示的小图标(可选)
3JZ9 G79H :+ AqY(Gz large-icon
(X|lK.W y 被可视化工具(诸如Jbuilder)用来显示的大图标(可选)
lmx'w 6z5?9I4[ description
Gu%}B@ 4^ 此tag标签的描述
m</nOf+C 51puR8AG> variable
`%.x0~ih 提供脚本变量的信息(同tei-class)(可选)
rlkg.e6 G:;(, attribute
^ %x7: Tag标签的属性名
#.<(/D+ 74a>}+" 以下章节介绍对于不同类型的tag,如何具体地实现它们。
A"T*uv| r%II`
i sX]ru^F3 简单的tag
&uxwz@RC0 ea!Znld] $c24l J#/ tag处理类
fYgX|#Me 5OX5\#Ux 简单的tag处理类必须实现Tag接口的doStartTag和doEndTag方法。当jsp引擎碰到tag标签的开头时,doStartTag被调用,因为简单的tag没有body,所以此方法将返回 SKIP_BODY。当jsp引擎碰到tag标签的结尾时,doEndTag被调用,如果余下的页面还要被计算,那它将返回EVAL_PAGE,否则将会返回SKIP_PAGE。
vLh,dzuo /N`E4bKBR 以下是例子:对于标签 <tt:simple /> ,它的tag处理类实现如下:
/-1 F9 m>-(c=3 TY gn
X public SimpleTag extends TagSupport
}t.VH:02y !e:HE/&>i {
q/t~`pH3 ]73BJ public int doStartTag() throws JspException
IF.6sJg: FrD,)Ad8Q {
\S"YLRn" j!QP>AM|` try{
!_=3Dz `K\(I#z pageContext.getOut().print(“Hello.”);
oS_<;Fj s~Od(,K }catch(Exception e){
a
|+q:g0M *dm?,~f%< throw new JspTagException(“SimpleTag: “ + e.getMessage());
U^{'"x+ -4'yC_8t }
;&G8e*bM2 4).>b3OhX return SKIP_BODY;
\%N |
X QH_I<Y:n }
bol#[_~ }w8:`g'T0/ public int doEndTag()
rocG;$[ VC@{cVT {
NHL{.8L{ >E`p@
e+ return EVAL_PAGE;
eb>YvC DEkFmmw
}
, *A', A9HJWKO }
-(fvb -O-qEQd *PjW, 注意:如果tag标签没有内容的话,那必须定义body-content元素为空,例如
YaDr.?
NuqWezJm& <body-content>empty</body-content>
^N5BJ'[F: ) b
vZ~t+^ a}+7MEUmZ/ 带属性的tag标签