在线时间 1957 小时 最后登录 2024-6-29 注册时间 2004-4-26 听众数 49 收听数 0 能力 60 分 体力 40957 点 威望 6 点 阅读权限 255 积分 23862 相册 0 日志 0 记录 0 帖子 20501 主题 18182 精华 5 分享 0 好友 140
TA的每日心情 奋斗 2024-6-23 05:14
签到天数: 1043 天
[LV.10]以坛为家III
群组 : 万里江山
群组 : sas讨论小组
群组 : 长盛证券理财有限公司
群组 : C 语言讨论组
群组 : Matlab讨论组
<b>构建含有数学内容的动态网站
2 [/ [7 n4 N& Q; ^% r </b>JavaServer Pages 技术和 LaTeX 帮助的联机科学教育与研究
) r' d7 ^/ y& q/ L# i
8 z+ q7 S! Q0 I& O1 |% _ < ><a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#author1" target="_blank" ><FONT color=#000000>Michael Juntao Yuan</FONT></A><A herf="mailtjuntao@astro.as.utexas.edu">juntao@astro.as.utexas.edu</A>)
1 I' A' B3 T/ X 博士研究生,德州大学奥斯汀分校
0 H, M$ L- G Z' ` 2002 年 2 月</P>
# J; H& j+ U. ~. } f5 @ <BLOCKQUOTE>将含有下标、上标和特殊符号的数学公式转换成 HTML 非常困难。大多数科学家和数学家使用灵活的 LaTeX 文本处理器来创建外表美观的公式 — 但是如何将这个输出转换成 Web 友好的格式呢?在本文中,Michael Yuan 说明了如何使用 JSP 页面和定制标记将 LaTeX 格式的公式放到 Web 上,这种方法对于编写者来说很简便并且不需要任何特定的客户机端软件。正如他在各种日益有效的方法中反复说明的,您会发现服务器方 Java 技术如何利用不同的第三方工具来解决看似很难处理的问题。通过单击本文顶部或底部的<B>讨论</B>在<A>论坛</A>中将您对本文的想法与作者和其他读者一起分享。</BLOCKQUOTE>( a, {- t$ t& A
< >因特网和 Web 最初是由科学家设计用来交换科学和数学研究信息的。出乎意料的是,经过因特网 30 多年的发展以及 Web 10 年的发展,仍然没有一种简便的方法将数学密集型内容发布到 Web 上。已经开发了 Web 数学通信标准,例如 W3C 的 MathML 建议;但是为了使用它们,数学内容的作者和读者都需要特定软件,比如用于 Netscape Navigator 和 Internet Explorer 的 IBM techexplorer 数学浏览器插件。(请参阅<a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#resources" target="_blank" ><FONT color=#000000>参考资料</FONT></A>以获取到 TechExplorer 和 MathML 的链接。)由于技术困难、成本和安全性危险因素,用户不太愿安装特定客户机端软件。因此,使那些标准的采用被限制在狭窄的研究领域,并且广大因特网用户无法使用它们。当前在 Web 上共享数学信息最方便的方法仍是将所有非文本公式转换成快照图像,然后将那些图像插入标准 HTML 页面。</P>! _: N; E& W# {2 |" E: W
< >最新的基于 GUI 的数学权威性工具具有将文档保存为“Web 页面”的选项;这些工具为您生成公式图像和静态 HTML 页面。如果您只是想在 Web 上发布静态内容,例如论文,则这种方法是很方便的。但是,静态网站的时代已经早就过去了,因为在大多数网站中用户的参与已经变得不可缺少。高级数学网站应该从后台数据库中动态地发布内容并允许用户提交他们自己的内容以与其他用户分享(示例包括交互式考试系统和研究协作系统)。</P>
1 r& {: S1 j, r7 l < >本文将向您展示该问题的一种解决方案。样本代码提供了一种方法,它可以快速简便地将表示数学公式的标记文本转换成显示该公式的可用于 Web 的图形文件。在此过程中,您会看到解决更多一般性难题的方法:我们将使用 JSP 页面和定制标记将几个完全不同的且互不相连的工具链接在单个程序中。当您使用多变的动态 Web 应用程序时,请记住不必从头做起;解决方案也可以其它形式提供,这由您控制。</P>
! ]- T, z+ g4 p: m < ><A>JSP 技术回顾</A>1 G4 E2 R4 n0 R' L; R
实现动态服务器方应用程序的领先技术是 JavaServer Pages(JSP)技术。可以使用 JSP 页面来控制如何在前端 Web 页面(显示逻辑)上显示后台数据。JSP 页面使用类似 HTML 的标记语言将 Java 数据对象与 HTML 显示元素粘合在一起。</P>
, |1 s5 h# Q7 J < >JSP 技术的主要优点是可扩展性。可以使用定制标记来扩展 JSP 标记语言,它允许我们封装在标记处理器类内部处理显示逻辑的 Java 代码。Web 编写者可以直接使用预定义的标记库完成复杂的显示任务,而不需要编写任何 Java 代码。定制标记使重用显示控制码成为可能,并且极大地简化了 Web 页面编写者的工作。常用的 JSP 定制标记类型是<I>内容转换</I>标记,它将包含在标记主体内对编写者友好的内容转换成对访问者友好的显示元素。</P>9 B; Z$ X$ t. \6 N& @- j0 ?
< >对于数学网站,可以使用内容转换标记将数学语义表示转换成 Web 图像。我们将在接下来的几节中更详细地讨论这一技术。</P>
S4 d0 R" m1 P < ><A>预备知识</A>) A' W8 Z6 F( }9 U2 V# m
这里说明了内容转换标记是如何工作的。当 JSP 容器遇到定制标记时,它在当前页的标记库描述符文件中查找标记名称。如果它找到一个匹配项,则 JSP 容器将上下文信息传递到标记处理器类。JSP 标记处理器首先从定制标记的主体中抽取表示数学公式的文本字符串。然后处理器根据语义文本公式创建图像,接着将该图像的 HTML 标记(<CODE><IMG></CODE>)发送到浏览器。在标记处理器中所需的数学图像生成工具应该具有下列功能部件:</P> u' g' _+ S! q7 Q% {, i: m9 s( j* u y
<UL>% N L! c0 G5 I2 x4 v9 {( y
<LI>必须用纯文本表示数学公式,纯文本是 JSP 标记主体中唯一支持的格式。纯文本还是通过 HTTP 协议传送信息和在数据库或 XML 文档中存储信息的标准格式。( w& O* w K3 Q5 _
# x, P3 r- k* S/ _5 Y* \
2 a0 M% K4 N$ P6 g 8 c2 u, Z; x" m. V
<LI>数学公式的纯文本表示法必须符合语义并且是直观的,这样 Web 编写者可以在无需经过许多训练的情况下使用它。为了实现该目的,如果全世界的科学家都已使用语义数学表示法体系,那么这是最好的。
6 x. g/ {& A* K" o7 {) } / ?- h, E5 x' g
1 K/ ?# q% B Q# T: V0 n% T 9 x* b+ n# k0 V! R6 a
<LI>数学显示工具在许多平台上应该都可用来发挥 Java 应用程序的跨平台性能。</LI></UL>3 Z( _. z0 X. X B2 b1 e2 O! e
< ><A>LaTeX 简介</A>
- n* j: w" i! A% h 符合以上所有要求的数学编写软件是一套业界标准的 TeX/LaTeX 类型设置工具(此后简称为 LaTeX)。LaTeX 提供下列功能:</P>
) ?" a' Y" Y5 }$ s7 Z; m <UL>
$ O5 e; F" B% V' z2 S- ` D& Q! d <LI>LaTeX 使用标准文本输入文件和一种完全开放的格式。事实上,大多数 LaTeX 实现也是开放源码的。
( }$ Q5 d; k" h! J4 o 4 V' ]1 [% Y/ V9 J; b9 c! ^
1 M/ C9 G: H' g6 [, M9 V 2 B$ }2 G" J/ j) J0 l: |
<LI>由于 LaTeX 的语法非常直观,所以很容易学习。例如,下列输入文本
8 c" x( Q4 r2 A0 ]+ K ) \7 y. H$ G: ~
3 M9 X; Y0 @3 q1 l . I2 {* p* b( y; G S3 p% E
<TABLE cellSpacing=0 cellPadding=5 width="80%" bgColor=#cccccc border=1>
, N+ S+ a$ D- [* o8 E* G$ q9 E 9 N' Y6 O, C- n
<TR>- n' H6 [4 b9 @5 P) M/ h
<TD>< RE><CODE> \alpha_0^2 + \beta_0^2 = \gamma_0^2 </CODE></PRE></TD></TR></TABLE>
" D" e5 @% \9 i) U8 [6 a + u- b& X$ o8 ?
& x, b) K4 M4 p2 N# H: x2 o$ m
4 G0 B9 t0 N7 ^3 C
< >通过 LaTeX 处理后生成图 1 中显示的公式。
% @) j. e" I. H, k
) n+ S, i/ w; E; P </P>4 M8 w2 }6 \1 G: P# F2 C
< ><A><B>图 1. 样本 LaTeX 输出</B></A><IMG src="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/fig01.gif"></P>
; E- `! m* [$ V. n: \0 H
& p* S. {" D, B ; n. s! E d, A9 Z
2 M; T! |9 l; g z1 h. n <LI>LaTeX 已经建立了一个很好的用户库,因为它在科学界中成为一种标准的类型设置语言已有一段时间了。大多数数学作者都已经熟悉它了。
% S- |1 y6 x1 ~' g
3 C4 V/ p" u1 c# r2 ~% }, I& ^+ G 1 m3 ?" [0 u/ w% S
3 r. j# U, U7 W3 d) W$ Q1 z) b
<LI>在很多平台上都可以使用 LaTeX 处理器,包括最常用的 UNIX、Linux、Windows 和 Mac OS。因此 LaTeX 文档有很高的可移植性。
5 i' u+ \% `) P
; v1 I/ X! c! U9 _. Z* J6 ^+ D% D/ w
$ q2 x- a0 A# S! X* ?( Q- j
1 G" T/ l0 r) ~, \# j0 E <LI>LaTeX 生成专业的具有出版质量的公式。这已经超出了 Web 图形的要求。图 2 显示使用 LaTeX 后呈现的麦克斯威尔方程组。
, l. `2 P Q' s - c' ~% R* u9 o7 g
! N* f6 e6 ]' I: z# t
# ?9 V5 }0 {# o0 F5 b6 L
< ><A><B>图 2. 使用 LaTeX 呈现的麦克斯威尔方程组</B></A><IMG src="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/fig02.gif"></P></LI></UL>
" Q" W# C' z% z* N( N/ `& p < >在继续阅读本文之前,如果对 LaTeX 及其语法感到陌生,最好熟悉一下。在下面的<a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#resources" target="_blank" ><FONT color=#000000>参考资料</FONT></A>一节中包含了到有用信息的链接。</P>: Y# ?( ~7 W' v$ L' J N+ F
< ><A>从 LaTeX 中生成 GIF 图像</A>
' o) M/ }& P( ?! D3 ` LaTeX 可以将语义数学公式处理成图像,但是无法将那些图像直接显示在大多数 Web 浏览器中。所有 Web 浏览器都支持 GIF 图像,因此我们需要将 LaTeX 公式图像转换成 GIF 格式。在本节中,我们将研究如何使用第三方工具从 LaTeX 输入文本中生成 GIF 图像。我们的最终目的是尽可能地使这个过程自动化,但是研究一下我们必须使用的基本构件块也是有所帮助的。我们将使用 Linux/UNIX 命令行工具作为示例,但是具有相同功能的工具也可以在 Windows 和 Mac OS 上使用。可以类似地生成其它格式的图像,例如,PNG 和 JPEG。下列步骤将创建 GIF 图像格式的数学内容:</P>- F& L1 l9 ^8 j
<OL>
3 w- S7 y0 C& D. o2 {1 e <LI>遵循 LaTeX 语法在文本输入文件中创建数学内容(我们把样本文件称为 sample.tex)。5 V* |6 {3 o$ ^- N# ^5 d4 g
4 d) E8 }9 R8 F* Y, t2 h
; ], b4 \5 j3 Y* d
<LI>使用图 3 中显示的 LaTeX 程序将 sample.tex 处理成与设备无关的文件(sample.dvi)。命令是 <CODE>latex sample.tex</CODE>。在该过程中,LaTeX 还生成一个日志文件和一个辅助文件。
7 v- Z1 U0 l( v: F. H) P
( y$ B. ?( l0 _; ?. {1 k2 @- u6 R
/ i+ c" `. f. C& g < ><A><B>图 3. 运行 LaTeX</B></A><IMG src="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/fig03.gif"></P>2 K/ K8 `4 U7 G! }) E
<LI>使用图 4 中显示的 Dvips 实用程序将 .dvi 文件转换成 PostScript 文件。命令是 <CODE>dvips -o sample.ps sample.dvi</CODE>。(有关 Dvips 的更多信息,请参阅<a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#resources" target="_blank" ><FONT color=#000000>参考资料</FONT></A>。)
0 f( q$ F/ p/ \6 f) r; p " H+ p6 B/ R" {7 H+ S0 x& d
$ M, h2 T4 U$ e0 @; i
< ><A><B>图 4. 运行 Dvips</B></A><IMG src="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/fig04.gif"></P>! }/ x; V" i! z* ~4 f, u
<LI>使用 ImageMagick 工具 <CODE>convert</CODE> 将 PostScript 文件 sample.ps 转换成 GIF 格式。(有关 ImageMagick 的更多信息,请参阅<a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#resources" target="_blank" ><FONT color=#000000>参考资料</FONT></A>。)命令是 <CODE>convert -antialias -crop 0x0 sample.ps sample.gif</CODE>。我们不想使 PostScript 页面上公式周围的空白部分变成 GIF 的一部分,因此我们使用 <CODE>-crop 0x0</CODE> 命令行选项来切除周围的背景。我们还使用 <CODE>-antialias</CODE> 选项来创建 antialias 字体。0 V2 ~0 |) Y7 J' d% P% f
! n6 l0 X1 u( T. M6 ^ N/ U( i
$ x9 `8 x2 [8 D" g+ l7 e
< ><A><B>图 5. 运行 ImageMagick</B></A><IMG src="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/fig05.gif"></P>
" z9 z% z0 ^# d' Z0 N W, K
0 W% ~! c: }. F; X/ u) o 您在命令行上告诉 ImageMagick 加到输出文件名上的文件扩展名确定输出图像格式。在上面的示例中,将输出图像指定为 <CODE>sample.gif</CODE>,它指示 ImageMagick 输出 GIF 图像。请参阅标题为“<a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#sidebar2" target="_blank" ><FONT color=#000000>矢量图形与位图的对比</FONT></A>”的边注,以获取有关 PostScript 到 GIF 转换的更多详细信息。</LI></OL>
; J. Y/ m2 c" P* X" `. V < ><A>在 Java 应用程序中运行外部程序</A>9 K/ H6 w4 g3 t2 J& j& G
到目前为止我们所列出的过程涉及了许多分离的命令行程序。要简便地将数学图像放到 Web 上,我们需要将那些进行数学编写的工具集成到 JSP Web 应用程序中。Java 程序可以使用 <CODE>java.lang.Runtime</CODE> API 来启动和管理 JVM 外部的主机 OS 进程。可以用类似下面的方法来启动新的进程:</P>! C& O! K4 F2 A, B: j
< >
/ n/ `1 S$ y3 ]6 Z% u; o <TABLE cellSpacing=0 cellPadding=5 width="65%" bgColor=#cccccc border=1>; z0 t) m$ M) n0 O. x2 O: B
* S$ s; d4 T# M <TR>
E1 Z+ H: j% Q) j# W$ y <TD>< RE><CODE> rocess p = Runtime.getRuntime().exec(cmd, envp, dir); </CODE></PRE></TD></TR></TABLE></P>
, k% R1 O+ u, S5 a+ ^5 |0 \6 r < >其中,参数 <CODE>cmd</CODE> 不是包含命令的单个字符串就是包含命令及其选项的字符串数组;字符串数组 <CODE>envp</CODE> 包含 <CODE>name=value</CODE> 格式的环境变量设置;参数 <CODE>dir</CODE> 是 <CODE>java.io.File</CODE> 类型并且它表示进程的工作目录。进程启动后,我们就可以将数据通过进程的输入流发送给它。</P>
! C% a8 O! L/ @9 b* z0 [ < >
5 E8 X3 b: ~/ ~2 G/ ~1 h. o <TABLE cellSpacing=0 cellPadding=5 width="65%" bgColor=#cccccc border=1>, ?8 K/ V6 O5 S" J) ]* m4 T6 T/ z# n
4 T2 J5 q" E& G4 p% ^& S <TR>3 M9 ?0 G, m0 k- ~
<TD>< RE><CODE>InputStream ins = p.getInputStream();</CODE></PRE></TD></TR></TABLE></P>
1 d: S9 H7 }4 A |6 y3 t < >我们还可以从进程的输出流中获取其输出:</P>6 l: z6 c6 d4 _7 N5 n
< >) O8 v& `$ o; k" K8 T ]
<TABLE cellSpacing=0 cellPadding=5 width="65%" bgColor=#cccccc border=1>( I2 B2 v: s; W& W
+ Z5 p* p4 g. R5 c% t
<TR>. O0 o. I# Y; a# w p1 h
<TD>< RE><CODE>OutputStream ous = p.getOutputStream();</CODE></PRE></TD></TR></TABLE></P>. G, X8 A8 C" u5 ^5 R
< >如果想暂停 Java 应用程序直到完成外部进程为止,可以使用下列代码:</P>/ N+ ?, J) z+ }5 ^1 v% I
< >
3 p5 a6 L! ^- a8 Y <TABLE cellSpacing=0 cellPadding=5 width="65%" bgColor=#cccccc border=1>
, H- _; M1 D6 O( d# o
$ Z* ]1 h. m5 ]7 W+ C+ q <TR>
3 I: w: @/ i. h. r& J4 D <TD><PRE><CODE>p.waitFor(); // Move on to other tasks. </CODE></PRE></TD></TR></TABLE></P>$ ~3 Y# k/ x5 V: y
<P><A>示例</A>
8 r9 u" ?8 h# p( Y) [, K2 P, l" O 我们讨论了实现含有动态数学内容的网站所需的工具和基础技术。现在,我们需要将它们放在一起。说明该方法的最好方式是运行与本文一起提供的示例应用程序。</P>
& c2 F! h. W- n2 c <P><A>如何部署示例</A></P>
! c# l1 O' O7 A <OL>
& z8 s; x6 M/ N <LI>下载并展开代码的 <a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/JSPMathCode.zip" target="_blank" ><FONT color=#000000>zip 归档文件</FONT></A>。" F; M/ k7 i1 U4 }/ l: k8 N; W
0 c. [8 Y" V6 m4 J
! H1 K/ P1 R/ _$ k3 i/ O <LI>编译 <CODE>support</CODE>、<CODE>SlowMathTag</CODE> 和 <CODE>MathTag</CODE> 包。您将需要在类路径中包含 J2EE v1.3 <CODE>j2ee.jar</CODE> 文件。
9 P" g) M: Q1 Y2 u) ~ ( ^9 x6 ^- D" u: a% y% O0 E
; Y5 f9 T; i. P% z; G! E( C
<LI>在 JSP 服务器中创建一个新的 Web 应用程序。例如,使用 Tomcat 服务器为测试应用程序创建目录 <CODE>Webapps/JSPMath</CODE>。4 H& p9 t) X% J
6 {. G+ b5 C4 k {( Z
- C- b: G! b5 V& V. F
<LI>将已编译包 <CODE>support</CODE>、<CODE>SlowMathTag</CODE> 和 <CODE>MathTag</CODE> 复制到 <CODE>Webapps/JSPMath/WEB-INF/classes</CODE> 目录下。( f/ P# @1 f3 ~* t8 P# A
$ c- R, V Q$ k, t6 o
- V+ |0 ^1 } U7 N' Z1 u3 T, U# b <LI>将标记库描述符文件 <CODE>tlds/*.tld</CODE> 复制到 <CODE>Webapps/JSPMath/WEB-INF/tlds</CODE> 目录下。: H& g4 m2 M+ O3 q K( `* J# d9 c* L _
- {1 i4 V: t( X$ L/ c
/ @) A7 M7 {' J. E- h- P% I, \ <LI>将 JSP 和 HTML 文件 <CODE>jsp/*</CODE> 复制到 <CODE>Webapps/JSPMath</CODE> 内的任何目录下。1 E% P& }, M( V4 u1 [% a; s& Y
0 R* _% J$ Q2 Q
$ D) R! U7 R7 Q$ b+ ^
<LI>现在,可以通过将浏览器指向 <CODE>Test01.html</CODE> 或 <CODE>Test02.html</CODE> 来开始测试。</LI></OL>
4 P" |7 z) J' r5 ] <P>一旦安装了应用程序,就可以遵循本节中的步骤来查看示例是如何工作的。</P>6 b4 [, p1 [2 o1 L; z% D3 N8 q
<P><A>正在运行的示例代码</A>
: N ~0 w j2 { 样本代码包含两种新的 JSP 定制标记。<CODE><LatexRaw></CODE> 标记将原始 LaTeX 文本转换成 HTML 图像标记。可以使用它在单个图像中显示多个公式、表或简短文本标题。<CODE><LatexMath></CODE> 标记只将单个公式图像的数学方式 LaTeX 文本转换成 HTML 图像标记。可以使用它来实现公式的快速显示。</P>% G# I9 i+ ~9 S. o9 `
<P>每个标记都有两个属性,<CODE>height</CODE> 和 <CODE>width</CODE>,它们对应于结果 <CODE><IMG></CODE> 标记中的属性 <CODE>height</CODE> 和 <CODE>width</CODE>。这些属性不是必需的。</P>
1 h% j; q/ G+ g+ Z. {; Q <P>下面的屏幕快照显示了正在运行的定制标记。Test01.html 在两个大文本域中采集 LaTeX 文本字符串。该页面顶部的两个小的文本输入框采集用户期望的其中一个结果图像的高度和宽度。在 Test01.jsp 中,我们将以这里指定的用户期望尺寸和由 ImageMagick 指定的缺省尺寸显示图像,以演示如何在定制标记中使用尺寸属性。已填写的 HTML 表单出现在浏览器中,如图 6 所示。</P>! i/ E9 _. e3 D7 u. ~
<P><A><B>图 6. LaTeX Math 输入表单</B></A><IMG src="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/fig06.gif"></P>
; ?0 K5 u! D7 `) I0 B3 R; W' ` <P>当我们在 Test01.html 中提交表单时,它将数据张贴到 JSP 页面 Test01.jsp 上,该页面显示了生成的公式图像,如图 7 所示。第一个图像是由标记 <CODE><LatexMath></CODE> 使用来自 HTML 输入表单中 <CODE>MathText</CODE> 域(单行文本域)的数学方式 LaTeX 文本生成的。请注意,第一个图像的 <CODE><LatexMath></CODE> 标记具有两个属性。然后按照用户指定的 <CODE>height</CODE> 和 <CODE>width</CODE> 属性的尺寸来显示图像。按照与第一个图像相同的方法来生成第二个图像,只是以缺省尺寸来显示第二个图像(生成第二个图像的标记没有尺寸属性)。第三个图像是混合文本和数学符号的示例,它由从 <CODE>RawMathText</CODE> 输入域(大型多行文本域)中的 LaTeX 文本的标记 <CODE><LatexRaw></CODE> 生成。指定的高度和宽度域不适用于这个图像。</P>
% k/ u, K* h- l: f' \ <P><A><B>图 7. 数学图像输出</B></A><IMG src="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/fig07.gif"></P>
% F& G+ v/ v4 H: b <P>清单 1 显示了 Test01.jsp 中的代码,它们可以实现以上所有目标。</P>
- V, r; j v$ _/ y/ {' r) n- X <P><A><B>清单 1. Test01.jsp</B></A>
4 a7 t1 d S( C! S @$ I <TABLE cellSpacing=0 cellPadding=5 width="100%" bgColor=#cccccc border=1>
# N" i* f5 c0 @5 A. l+ G9 }& \) ^
; E3 z4 N6 A: W7 c: X1 y( z, X# v <TR>' w+ [8 i; {6 R1 T
<TD><PRE><CODE><%@ taglib uri="/WEB-INF/tlds/SlowMathTaglib.tld" prefix="SlowMathTag" %><% String mathText = request.getParameter( "MathText" ); String rawMathText = request.getParameter( "RawMathText" ); String height = request.getParameter( "height" ); String width = request.getParameter( "width" );%><html><head> <title>JSPMath test without image cache</title></head><body bgcolor=white><p><font color=green> Math formula in desired dimensions: </font><p><SlowMathTag atexMath height="<%=height%>" width="<%=width%>"> <%=mathText%></SlowMathTag atexMath><p><font color=green> Math formula in default dimensions: </font><p><SlowMathTag atexMath> <%=mathText%></SlowMathTag atexMath><p><font color=green> Latex segment in default dimensions: </font><p><SlowMathTag atexRaw> <%=rawMathText%></SlowMathTag atexRaw></body></html></CODE></PRE></TD></TR></TABLE></P>
" f' v1 }/ \- T3 }7 w3 [ <P>清单 2 显示了在一次执行中 JSP 容器发送到 Web 浏览器的实际 HTML 代码。</P>5 x) k6 R3 H7 ?+ v# _4 y
<P><A><B>清单 2. 从 Test01.jsp 发送到浏览器的 HTML 代码</B></A>
- o7 K" [9 P) p6 _: _; F <TABLE cellSpacing=0 cellPadding=5 width="100%" bgColor=#cccccc border=1>: k$ s6 r( \3 b- u# ?3 @' u! n
& x. H# s5 Y0 L4 A' z <TR>
, ^0 x8 c2 `9 H! ?$ ~2 ] <TD><PRE><CODE><html><head> <title>JSPMath test without image cache</title></head><body bgcolor=white><p><font color=green> Math formula in desired dimensions: </font><p><IMG SRC="http://localhost:8080/JSPMath/images/image1009709773390.gif" height=20 width=150 /><p><font color=green> Math formula in default dimensions: </font><p><IMG SRC="http://localhost:8080/JSPMath/images/image1009709776535.gif" /><p><font color=green> Latex segment in default dimensions: </font><p><IMG SRC="http://localhost:8080/JSPMath/images/image1009709777612.gif" /></body></html></CODE></PRE></TD></TR></TABLE></P>5 U$ M& y: T9 Q
<P>这里的 GIF 图像是由标记处理器生成的。如图 7 所示,使用用户指定的尺寸放大的图像看起来不是很平滑。有关出现这一情况的原因以及如何在 UNIX/Linux 系统上处理它,请参考标题为“<a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#sidebar2" target="_blank" ><FONT color=#000000>矢量图与位图的对比</FONT></A>”的边注。</P>
: _: J# |8 a/ c0 o) o8 O* g2 s+ _ <P>我们如何实现这些定制标记操作呢?请继续阅读,找出答案。</P>
" H0 s9 X$ t4 ^2 e <P><A>一种简单(但缓慢)的方法</A>
4 ^2 V# Y/ A) s) X; D. L 可以用很简单和直接的方式实现上面的示例。对于本文,您应该已经知道如何实现 JSP 定制标记处理器了。如果您想更新一下您的知识,请参考下面的<a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#resources" target="_blank" ><FONT color=#000000>参考资料</FONT></A>一节。</P>$ N5 T. E' u% x5 P- l
<P><CODE><SlowMathTag:*></CODE> 标记处理器调用静态方法 <CODE>support.MathUtil.latex2gif()</CODE> 以生成 GIF 图像。让我们研究一下这种方法的源代码以及它是如何工作的。</P>2 y7 n- ?: X: \" e8 v' E
<P>首先,该方法确定当前标记生成的 GIF 图像以及中间文件的唯一名称,如清单 3 所示。那些文件的根名称是当前时间的毫秒值。</P>0 v! C# O5 d# q# L7 Q) s) G" A5 M
<P><A><B>清单 3. 生成文件名</B></A> : h: w3 a8 D! p
<TABLE cellSpacing=0 cellPadding=5 width="100%" bgColor=#cccccc border=1>- h6 n3 f4 j6 h. N, V* i Z
$ L/ Z& c0 \# D+ J
<TR> f. Q/ C2 r6 O# t
<TD><PRE><CODE>Date now = new Date ();String filename = "image" + Long.toString(now.getTime());// ...File f = new File(path + filename + ".tex");</CODE></PRE></TD></TR></TABLE></P>0 w, t1 ~! r4 ^8 \! D$ t1 e% r
<P>接下来,该方法从标记主体中采集内容,然后组装成一个完整的 LaTeX 输入文件,如清单 4 所示。JSP 容器用字符串变量 <CODE>latexStr</CODE> 将标记主体传送到方法 <CODE>latex2gif()</CODE>。使用 <CODE>\pagestyle{empty}</CODE> 命令以消除在结果页面底部显示的页号。使用 <CODE>\batchmode</CODE> 命令使 LaTeX 处理器在非交互方式下执行。</P>' U& Y2 Y* H- M9 t. x, P
<P><A><B>清单 4. 汇编 LaTeX 输入文件</B></A>
" Q I* g n" G <TABLE cellSpacing=0 cellPadding=5 width="100%" bgColor=#cccccc border=1>
) c5 E5 b* f! Q9 u( E% ~2 u
$ j, R" W* l' M$ G( ~$ C <TR>
; y' B, B! B1 @# I <TD><PRE><CODE>private static String latexHead = "\\batchmode\n" + "\\documentclass[12pt]{article}\n" + "\\pagestyle{empty}\n" + "\\begin{document}\n";private static String latexEnd = "\\end{document}\n";// ...// String variable "latexStr" contains the body of the current tag.// Process modes.if ( mode.equals("displaymath") ) { latexStr = "\\Large\n\\begin{displaymath}\n" + latexStr.trim() + "\\end{displaymath}\n";} else { // Unrecognized modes are raw latex strings.}// Write content into a latex file.FileWriter fw = new FileWriter(f);fw.write( latexHead, 0, latexHead.length() );fw.write( latexStr, 0, latexStr.length() );fw.write( latexEnd, 0, latexEnd.length() );</CODE></PRE></TD></TR></TABLE></P>
8 }. J, W9 o3 D$ s9 Z <P>现在,该方法执行命令 <CODE>latex</CODE>、<CODE>dvips</CODE> 和 <CODE>convert</CODE>,以在服务器方的预先配置目录 <CODE>path</CODE> 下生成最终的 GIF 图像,如清单 5 所示。</P>" i4 S/ S- M- l! F% v
<P><A><B>清单 5. 生成 GIF 文件</B></A> " x$ O& @$ ^; F6 G
<TABLE cellSpacing=0 cellPadding=5 width="100%" bgColor=#cccccc border=1>
4 S1 v/ e1 ^* R
- Y: e$ N0 t, F! V! H <TR>
, m1 v! \) |1 s0 y [( ?3 r& Y' o <TD><PRE><CODE>private static String latexCommand = "/usr/bin/latex";private static String dvipsCommand = "/usr/bin/dvips";private static String convertCommand = "/usr/X11R6/bin/convert -antialias -crop 0x0";// ...public static synchronized String latex2gif (String path, String mode, String latexStr) throws Exception { // Generate root file name and create ".tex" file // ... // Get Runtime. Runtime r = Runtime.getRuntime(); Process p; File imagePath = new File( path ); // ... // Run latex in directory "imagePath". p = r.exec(latexCommand + " " + filename + ".tex", null, imagePath); if ( p.waitFor() != 0 ) throw new Exception("Error in Latex\n"); // Run dvips in directory "imagePath". p = r.exec(dvipsCommand + " -o " + filename + ".ps " + filename + ".dvi", null, imagePath); if ( p.waitFor() != 0 ) throw new Exception("Error in dvips\n"); // Run convert in directory "imagePath". p = r.exec(convertCommand + " " + filename + ".ps " + filename + ".gif", null, imagePath); if ( p.waitFor() != 0 ) throw new Exception("Error in convert\n"); // ...}</CODE></PRE></TD></TR></TABLE></P>
9 Y; _2 }# `. X <P>最后,该方法删除所有中间文件,如清单 6 所示。</P>' W, w. _+ F1 R9 z
<P><A><B>清单 6. 删除中间文件</B></A>
2 Q) W1 `% M' z" t <TABLE cellSpacing=0 cellPadding=5 width="100%" bgColor=#cccccc border=1>: L* \& E9 I+ q8 h6 e2 O; x/ d
5 W" g" Q1 u9 G9 g
<TR>
4 z; A3 u( V) o8 l <TD><PRE><CODE>f = new File(path + filename + ".tex");f.delete();f = new File(path + filename + ".log");f.delete();f = new File(path + filename + ".aux");f.delete();f = new File(path + filename + ".dvi");f.delete();f = new File(path + filename + ".ps");f.delete();</CODE></PRE></TD></TR></TABLE></P>
. I$ L5 Q9 `) Q <P>现在,标记处理器可以将生成的 GIF 图像的 <CODE><IMG></CODE> HTML 标记写入服务器输出流。</P>
! Y- Z, r/ |: `8 ~) X7 a8 G <P>刚刚描绘的方法很简单,但是正如我们马上就会看到的它很慢而且容易出错。在接下来的几节中,我们将讨论改进这一解决方案的性能和健壮性的方法。</P>
4 d% L) p4 I# n5 K) Y m) r# V <P><A>通过高速缓存改进性能</A>
! s9 s- o, P' W2 u! N 从 LaTeX 输入文本生成图像是很慢的,因为它涉及多个外部程序。在上一节的简单实现中,每次调用 <CODE><SlowMathTag:*></CODE> 定制标记处理器时,它都必须生成一个新图像。这样效率是非常低的。在示例应用程序中,Test01.jsp 中的两个 <CODE><SlowMathTag atexMath></CODE> 标记需要为每个访问者两次生成相同的公式图像,只是为了用不同的尺寸来显示它。现实世界中,应用程序很可能比示例应用程序复杂得多,并且生成不必要的图像带来的性能开销可能很严重。想象一下,我们的站点将提供动态数学考试,并且许多访问者会遇到相同的考试题目。为每个访问者一次次地重新生成同一公式的图像太昂贵了。</P>
4 B0 N# a, F* ?( w7 G; V8 v <P>对于一个规模可伸缩的解决方案,我们需要对生成的图像进行高速缓存,然后在将来需要的时候重用它们。可以将 LaTeX 文本用作图像名称的<I>键</I>。如果标记处理器第二次遇到同一个 LaTeX 文本,则它可以找出高速缓存图像的文件名,然后直接返回它。可以将一对对的 LaTeX 文本和图像名称存储在 SQL 数据库以便长久保存和快速搜索。可以使用具有不同作用域的数据库访问 JavaBean 来对那些高速缓存的图像的访问作进一步的调节。虽然对于读者来说,这样的调节可能是一个有趣的实践,但是却超出了本文的范围。</P>" ~1 b8 k4 ~; e0 r: D
<P>这里,我们将把高速缓存的图像作为静态数据成员存储在标记处理器助手类中。那些静态数据成员具有与 JSP 服务器容器相同的生命期。如果更改网站,然后重新启动服务器,则会丢弃旧的高速缓存。将高速缓存的 LaTeX 文本和图像名称存储在两个静态的 <CODE>Vector</CODE> 中。当标记处理器遇到 LaTeX 文本字符串时,它遍历高速缓存的 LaTeX 文本 <CODE>Vector</CODE>。如果它发现一个匹配项,则返回相应的图像名称。如果没发现,则继续生成一个新的公式图像并将新的 LaTeX 文本/图像对存储在高速缓存中。清单 7 说明了这个方法。</P> B5 }& {4 E g8 S( W" U
<P><A><B>清单 7. MathUtil.getImage()</B></A>
" S% T5 }. j8 |* }& {6 b <TABLE cellSpacing=0 cellPadding=5 width="100%" bgColor=#cccccc border=1>
4 E+ s6 Y, ?. H9 ^. d! j( k: q: g4 x 2 A2 p- X! _! h3 d$ i
<TR>! m8 w' M5 K9 a* f# {" ^
<TD><PRE><CODE>public static synchronized String getImage (String path, String mode, String latexStr) throws Exception { // Formatted string stored in the latexStrVec vector. String comp = "MODE=" + mode + "\nLaTeX=" + latexStr.trim(); // Iterate through the "latexStrVec" cache and check if any // element matches the input string and mode. for (int i = 0; i < latexStrVec.size(); i++) { String l = (String) latexStrVec.elementAt(i); if ( comp.equals(l) ) { // Return the cached image name if a match is found. return (String) imageNameVec.elementAt(i); } } // If no match is found, generate a new image // and store it in the cache. return storeImage(mode, latexStr, latex2gif(path, mode, latexStr) );}</CODE></PRE></TD></TR></TABLE></P>0 L* q2 \) ?+ O, I, y# @4 l
<P>Test02.html 和 Test02.jsp 将高速缓存改良的标记处理器用于 <CODE><MathTag:*></CODE> 标记。尝试它们,然后在 <CODE><SlowMathTag:*></CODE> 标记上感受这一改进。</P>, @$ @# Q% p; ~" Q. u" p0 o3 B7 @
<P><A>减少中间文件</A>
3 }1 k( m0 K( j! E, Y: h& ] 正如我们所见,实用程序使用很慢的磁盘文件互相通信,并且我们需要在完成该过程时删除所有中间文件。那些中间文件不仅造成了不必要的开销还容易导致磁盘和文件系统错误。在 UNIX/Linux 系统中,可以使用管道连接不同程序的输入和输出,并且部分消除对中间文件的需要。因为 Java 运行时不直接支持 UNIX 管道,所以我们必须从 shell 调用它们,如清单 8 所示。</P>
; k- I4 U, X7 \- d5 ~ <P><A><B>清单 8. 使用管道输入/输出调用命令</B></A>
8 H: ^+ I' U( X2 w4 S9 O! I% X <TABLE cellSpacing=0 cellPadding=5 width="100%" bgColor=#cccccc border=1>
: x9 O4 U/ `, {: J* x
( I; C4 j. c9 W8 q0 b# X8 V* |) S <TR>1 K! Q6 d \- r; g2 a5 {5 B9 S
<TD><PRE><CODE>String convertCommand = "/usr/X11R6/bin/convert -antialias -crop 0x0";String dvipsCommand = "/usr/bin/dvips";// ...Process p = Runtime.getRuntime().exec( new String[] { "/bin/sh", "-c", dvipsCommand + " -f < " + filename + ".dvi | " + convertCommand + " - " + filename + ".gif" }, null, imagePath);</CODE></PRE></TD></TR></TABLE></P>- a8 L' t8 j) t2 J, P
<P>但是 LaTeX 程序不支持标准输入/输出管道,并且 Dvips 需要可搜索的输入(即,它不能是管道)。因此我们仍然需要 .tex 和 .dvi 中间文件。</P>
1 u1 ^; v2 M: M: w6 W& c# x! z, V <P>因为这个方法只适用于 UNIX,并且无法除去所有中间文件,所以我在示例程序中没有使用它。在下一节中,我们将讨论更一般和有效的方法。</P>
/ q& k# ?$ R9 G! i6 }7 T <P><A>用于外部应用程序的 Java API</A>
$ y9 \6 N% h8 E/ q" ] 象我们在上面所做的在独立 Java 运行时进程中调用单独的程序很方便。但是,这个方法不仅仅会导致我们已经遇到的中间文件问题;而且无法与复杂的 Java 应用程序一起很好地工作。我们无法容易地将外部程序集成到 Java 类分层体系结构中。</P>
& P3 d' Z' o1 r0 J9 J <P>如果我们可以直接从 Java 程序调用 LaTeX 和图像转换实用程序功能,并使用 Java 内存中的缓冲区来交换进程间的数据,则创建的代码可以更快且可维护性更高。这需要 Java 编程 API 来访问那些功能。</P>! P+ \2 W# f$ A. o( F: r# B
<P>为非 Java 应用程序生成 Java API 的一种标准方法是使用 Java 本地接口(JNI)。要使用 JNI,需要重新编译带有新的 Java 可识别的头文件的非 Java 应用程序。到目前为止提到的所有数学工具都来自开放源码项目,并且它们的源代码可用于重新编译。我们可以修改源代码的输入/输出部分让那些工具能够处理输入/输出缓冲区而不是文件。实际上,已经开发了 JMagick,它的目的是将一个 JNI Java API 添加到 ImageMagick 库(请参阅<a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#resources" target="_blank" ><FONT color=#000000>参考资料</FONT></A>)。为所有数学工具创建 JNI API 明显超出了本文的范围。但是,如果您正在考虑实现复杂 Java 应用程序的类似体系结构,则可能要考虑朝这个方向努力。有关 JNI 的更多信息,请参考下面<a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#resources" target="_blank" ><FONT color=#000000>参考资料</FONT></A>一节。</P>
3 E" @/ o) a: x: U) F) Y/ x' p <P><A>总结</A>' C2 @' V* ^1 }: e$ m3 T- F
在本文中,您已经了解了几种提高效率的技术,它们将使用 JSP 定制标记把 LaTeX 样式的文本转换成数学图像。可以很简便地扩展这个方法,为其它高级内容管理技术(例如 <a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/index.shtml#advanced" target="_blank" ><FONT color=#000000>XSLT</FONT></A>)创建 Java 扩展。应该考虑还有什么其它工具可用来修改它们,然后集成到 Web 项目中。使用这里提到的技术,我们现在拥有了一些能创建含有数学密集型内容的高级网站的工具。</P>: H3 |) t, ^/ S) N: e6 g* r
<P>我想知道您关于如何改进这一解决方案的意见和建议。请使用<A>论坛</A>让大家分享您的想法。</P>
9 }6 ]! [" e5 X! }8 n I% h <P><A>参考资料</A>
+ x+ U" D6 F T, u; { <UL>
0 @6 G% u, u3 ]1 j3 d/ ]4 m( g <LI>请通过单击本文顶部或底部的<B>讨论</B>来参与本文的<A>论坛</A>。2 Q& Q+ K" v/ O0 E* j) C
' L: }- T& x/ S4 f5 V6 y
" @/ U$ h9 R) `, ^ <LI>下载本文中示例项目的<a href="http://www-900.ibm.com/developerWorks/cn/java/j-jspmath/JSPMathCode.zip" target="_blank" ><FONT color=#000000>源代码</FONT></A>。</LI></UL>
6 S K% I8 ~7 i i <P>
# h$ U$ T# T" p& z {9 y) d <UL>8 n( R6 h( S0 u1 O$ o R8 K
<LI>有关 JSP 的背景知识,请查看 Noel J. Bergman 的教程“<a href="http://www-105.ibm.com/developerWorks/education.nsf/java-onlinecourse-bytitle/882707E838C672A185256770004BDE72?OpenDocument" target="_blank" ><FONT color=#000000>Introduction to JSP technology</FONT></A>”(developerWorks,2001 年 8 月)。
* `1 h6 t- l* ?9 I" ?8 ]. l 0 ^/ t0 B6 c9 N5 H' P
; a6 h! ]: A, E+ \& ` <LI>在“<a href="http://www-106.ibm.com/developerWorks/java/library/j-taglib/" target="_blank" ><FONT color=#000000>Take control of your JSP pages with custom tags</FONT></A>”一文中,Jeff K. Wilson 提供了定制标记库如何允许您向 JSP 添加更复杂逻辑、更稳定地控制数据显示以及在标记中共享数据的详细信息。
% N1 t V# H; e) f
9 W5 }) y/ d1 N/ N * \& g6 b; g9 F, x1 |3 Q. @
<LI><I><A herf="http://www.manning.com/fields2/">Web Development with JavaServer Pages, 2nd Edition</A> 一书(由 </I>D. K. Fields,M. A. Kolb 和 Shawn Bayern 合著)(Manning Publications,2001 年)是一本几乎包含了 JSP 各个方面的好书。
) T$ s- P. I3 Y( v. y" }$ ?
4 |* W6 V1 x: F* b4 E; p 9 h- Z$ B! D( ?8 I, [
<LI><I><A herf="http://www.manning.com/shachor/">JSP Tag Libraries</A>(</I>由 Gal Shachor、Adam Chace 和 Magnus Rydin 合著)(Manning Publications,2001 年)对于如何设计和实现 JSP 定制标记库有一些好的示例和解释。
% @+ p% J0 }3 S
) O1 \( F) v( {
+ X0 |& \9 I! V0 g* B* }8 } <LI>在“<a href="http://www-106.ibm.com/developerWorks/java/library/j-jsptags.html" target="_blank" ><FONT color=#000000>JSP taglibs: Better usability by design</FONT></A>”(developerWorks,2001 年 12 月)中 Noel J. Bergman 讨论了 JSP 定制标记设计问题。) d2 N" O2 y4 ^% i8 [2 U& p
/ j& [; c9 }0 j' {) E9 W5 D8 x! { * P+ X9 l) A2 p
<LI>WebSphere 开发者园地为那些想关注<a href="http://www-106.ibm.com/developerWorks/cgi-bin/click.cgi?url=http://www7b.boulder.ibm.com/wsdd/library/tutorials/vajwebsph353/Part-I/JSP11Part-I.html&origin=j" target="_blank" ><FONT color=#000000>将 JSP 定制标记和 VisualAge for Java 和 WebSphere Studio 组合起来</FONT></A>的人提供了一个很好的教程。</LI></UL>1 q( R( h) S% A' T/ B9 G5 e P! F
<P><B>工具</B></P>
; {. |" R6 H- V$ W" {0 V <UL>
$ h: S7 [# S+ a2 _: [0 S4 l <LI>研究一下 <a href="http://www.latex-project.org/" target="_blank" ><FONT color=#000000>LaTeX</FONT></A> 项目的主页。- A( K7 }. ? C: `- \: K2 ]
' u' b& Y- c1 t" E1 m R( {$ x ! @% L) q) w4 G1 b) O: N/ K- z( g
<LI>查看 <a href="http://www.imagemagick.org/" target="_blank" ><FONT color=#000000>ImageMagick</FONT></A> 图像操作工具。8 ]2 q8 z3 z, w8 D1 Z7 g/ u
5 q8 [0 W9 }3 L- ?$ D0 A( Z
2 K2 |3 ^+ U: ]$ z) N" M <LI><a href="http://www.yeo.nu:81/jmagick/" target="_blank" ><FONT color=#000000>JMagick</FONT></A> 一个 ImageMagick Java API 的工作正在进行。
. W9 }: \: w% {, u/ ~7 i+ @ 5 E- P6 A! z' d* n8 J# P1 U+ P
& q' `7 p6 w- q9 e3 ^! T; _! o
<LI><a href="http://www.fourmilab.ch/webtools/textogif/textogif.html" target="_blank" ><FONT color=#000000>Textogif</FONT></A> 将 TeX/PostScript 文件转换成任何分辨率的 GIF 图像。 ^; B: N& G! O$ c, l
% O) c4 W- R$ {2 R. f- Y5 P
& M8 k7 q( i9 T) p r1 S <LI><a href="http://java.sun.com/products/jdk/1.2/docs/guide/jni/" target="_blank" ><FONT color=#000000>Java Native Interface</FONT></A>。</LI></UL>
, C( e2 ]1 ~$ X. T0 m# w- @* y <P>8 k& D0 _1 l! G( D9 i' H- }
<UL>1 B+ |' b. c- b( P5 f
<LI><a href="http://www.w3.org/Math/" target="_blank" ><FONT color=#000000>MathML</FONT></A> 是 W3C 的标准数学标记语言。4 E& ^& ^! b" u2 }6 t( N
. E1 G5 g. o6 B" [1 j9 d" d
" w$ B% W% a- N8 U% p <LI>查看 IBM 的数学文档浏览器 <a href="http://www-3.ibm.com/software/network/techexplorer/" target="_blank" ><FONT color=#000000>techexplorer</FONT></A>。</LI></UL>
2 y# t; l5 K4 L4 V <P>( f# Y4 i7 w/ V" F, L9 K9 H
<UL>8 M$ k/ c3 c" G) f) D
<LI>Pramod Kankure 教您如何“<a href="http://www-106.ibm.com/developerWorks/library/x-dynweb/index.html" target="_blank" ><FONT color=#000000>Produce Dynamic Web Pages with Java and XSLT</FONT></A>”(developerWorks,2001 年 4 月)。
/ S) v4 s5 W: z
7 C+ L$ e0 ^% b0 O" z
0 o, \9 a W3 M! u1 E <LI>有关 Java 和 XSLT 的更多信息,请查看由 Taylor Cowan 所著的“<a href="http://www.javaworld.com/javaworld/jw-12-2001/jw-1221-xslt.html" target="_blank" ><FONT color=#000000>XSLT Blooms with Java</FONT></A>”(<I>JavaWorld,</I>2001 年 12 月)。</LI></UL>
( U4 X p4 k* L& z <P>+ f3 R! m# h% Q5 b8 {
<TABLE cellSpacing=0 cellPadding=0 width="100%" border=0>
: r# l8 h" O$ c' U. s: n 9 Z+ [! U9 ?( M0 z
<TR>
% g' s) @! E0 S5 Q, O <TD><A>关于作者/ L: g& \$ U/ S& G0 X
Michael Yuan 是德州大学奥斯汀分校的博士研究生。他对使用服务器方 Java 技术来促进科学教育和研究很感兴趣。可以通过 <a href="http://www.shumo.com/bbs/mailtjuntao@astro.as.utexas.edu" target="_blank" ><FONT color=#000000>juntao@astro.as.utexas.edu</FONT></A> 与他联系。</TD></TR></TABLE></P>
zan