Apache Cocoon and XSP的功能介绍
jc2cn 原创 ApacheCocoonandXSP (bigmouse@peoplemail.com.cn) ============================= 一.介绍 ApacheCocoon是一个纯Java的网页内容发布框架,它允许内容(纯数据)、逻辑和表现形式分别放到不同的XML文件中,然后用XSL来将它们组合并显示出来。这种结构的好处是,表现与内容无关,例如,存放在一个XML文件中的一本书的内容,可以根据表现形式的不同要求,用XSL转换成HTML格式、PDF格式甚至是WML(当然现在WAP不适合看书。。。呵)等不同的形式,而且由于XML的表现无关性,使之作为一种标准的传输交换数据格式在B2B(企业-企业)商务活动中以及分布式程序的传输交互(例如,SOAP)中有着广阔的应用。 XSP即ExtensibleServerPages的缩写,它通过在XML文件中加入Java程序来实现商务逻辑,由于XML文件中的数据是通过XSL来表现的,所以XSP做到了把内容与显示分开。想一想你熟悉的ASP、PHP、JSP是怎么做的?它们是把逻辑代码插入到HTML页面中来工作的,也就是说内容与显示并没有分开。这很容易造成麻烦,如果你用这些技术开发过网站就会有体会,必须先等美工人员给你做好了的HTML页面,然后你嵌入你的那些代码(如果做的够久,干脆Ctrl+CandCtrl+V)。以后美工人员想改点什么还要先抓住你问:“老兄,我动动这块儿没事吧?!”。更糟糕的是网站要改版呢? 在用Cocoon构建网站中,工作人员被分为三种:XML文件的编写者、XSP的编写者和XSL的编写者。XML文件的编写者主要是编写XML文件格式,DTD或者Schema,这工作相当于通常网站的内容编辑。XSP编写者就是负责往XML文件中加入Java逻辑代码,来动态控制内容,这工作相当于通常网站的ASP、PHP、JSP程序编写者。XSL编写者负责编写显示页面的XSL文件,也就是通常网站的美工人员,不过这比一般美工人员要求高,因为XSL由模板组成,就是一个个调用XML的相关元素而没有内容的空架子,这些模板也是有固定语法的,具体怎么回事以后部分能看到。 二.安装Cocoon 虽然JBuilder6开发环境可以调试CocoonWeb程序,但也许你并不使用JBuilder6,所以下面我介绍一下在Windows2000professional中的安装配置过程(以我机器上的目录为例): 1.JDK 我用的是JBuilder自带的JDK1.3.1,安装目录为c:/jbuilder6/jdk1.3.1/。如果你还没有JDK安装程序,请到http://java.sun.com去下载一份。 直接安装。 2.ApacheHTTPServer 我用的是PHPTriadforWindows自带的Apache1.3.12,安装目录为c:/apache/。如果你还没有Apache安装程序,请到http://www.apache.org去下载一份。 直接安装,注意修改c:/apache/conf/httpd.conf文件中的port项(大约第211行),如果你有其他HTTPServer比如IIS,请修改此端口值,不要跟IIS冲突。我的port值设定为8000。 3.Tomcat 我用的是JBuilder自带的Tomcat3.2.3,安装目录为c:/jbuilder6/jakarta-tomcat-3.2.3/。如果你还没有Tomcat安装程序,请到http://jakarta.apache.org去下载一份。 下载ApacheModuleJServ.dll(还是上面的那个网站),将文件拷贝到c:/apache/modules/目录中。 修改c:/jbuilder6/jakarta-tomcat-3.2.3/conf/tomcat.conf文件,将"LoadModulejserv_modulemodules/ApacheModuleJServ.dll"前面的"#"去掉(大约第8行),并且将"LoadModulejserv_modulelibexec/mod_jserv.so"前面加上"#"(大约第13行)。 修改c:/apache/conf/httpd.conf文件,在文件最后加上"includec:/jbuilder6/jakarta-tomcat-3.2.3/conf/tomcat.conf" 右键单击"我的电脑"->"高级"->"环境变量",在"系统变量"中"新建"下面两个环境变量:"变量名"为"TOMCAT_HOME","变量值"为"c:/jbuilder6/jakarta-tomcat-3.2.3";"变量名"为"JAVA_HOME","变量值"为"c:/jbuilder6/jdk1.3.1"。 4.Cocoon 我用的是JBuilder自带的Cocoon1.8,安装目录为c:/jbuilder6/cocoon/。如果你还没有Cocoon安装程序,请到http://xml.apache.org去下载一份。 将c:/jbuilder6/cocoon/lib/目录下的所有*.jar文件拷贝到c:/jbuilder6/jakarta-tomcat-3.2.3/lib/目录下面。 将c:/jbuilder6/cocoon/bin/目录下的cocoon.jar文件拷贝到c:/jbuilder6/jakarta-tomcat-3.2.3/lib/目录下面。 注意:我用的Tomcat版本可以自动检查c:/jbuilder6/jakarta-tomcat-3.2.3/lib/目录下的*.jar文件,然后加入到$CLASSPATH中,如果你发现你的Tomcat不支持自动检查功能,你要手动向tomcat.bat中加入那些拷贝过去的*.jar文件。 在c:/jbuilder6/jakarta-tomcat-3.2.3/webapps/下建立cocoon子目录,然后在c:/jbuilder6/jakarta-tomcat-3.2.3/webapps/cocoon/目录下建立WEB-INF子目录。 将c:/jbuilder6/cocoon/conf/cocoon.properties文件拷贝到c:/jbuilder6/jakarta-tomcat-3.2.3/webapps/cocoon/WEB-INF/中。 将c:/jbuilder6/cocoon/src/WEB-INF/web.xml文件拷贝到c:/jbuilder6/jakarta-tomcat-3.2.3/webapps/cocoon/WEB-INF/中。 修改c:/jbuilder6/jakarta-tomcat-3.2.3/webapps/cocoon/WEB-INF/web.xml文件,将其中的conf/cocoon.properties改成WEB-INF/cocoon.properties 修改c:/apache/conf/httpd.conf文件,在最后加上: Alias/cocoonc:/jbuilder6/jakarta-tomcat-3.2.3/webapps/cocoon <Directory"c:/jbuilder6/jakarta-tomcat-3.2.3/webapps/cocoon"> OptionsIndexesFollowSymLinks </Directory> ApJServMount/cocoon/cocoon <Location/cocoon/WEB-INF/> AllowOverrideNone denyfromall </Location> 修改c:/jbuilder6/jakarta-tomcat-3.2.3/conf/server.xml文件,加上: <Contextpath="/cocoon"docBase="webapps/cocoon"debug="0"reloadable="true"> </Context> 最后重新启动Tomcat和Apache,让设置生效。 在IE中访问:http://localhost:8080/cocoon/Cocoon.xml 可以看到Cocoon的一些参数。 三.用java写XSP的逻辑标签 先看一个XSP的例子,这是一个简单的计数器: ----------counter.xml---------- <?xmlversion="1.0"encoding="gb2312"?> <?cocoon-processtype="xsp"?> <?cocoon-processtype="xslt"?> <?xml-stylesheethref="counter.xsl"type="text/xsl"?> <xsp:pagelanguage="java"xmlns:xsp="http://www.apache.org/1999/XSP/Core"> <xsp:logic> privatestaticintnCounter=0; privatesynchronizedintgetCounter() { returnnCounter++; } </xsp:logic> <counter> <p> 访问:<xsp:expr>getCounter()</xsp:expr>(次) </p> </counter> </xsp:page> ----------counter.xsl---------- <?xmlversion="1.0"?> <xsl:stylesheetversion="1.0"xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> <xsl:templatematch="counter"> <xsl:copy> <xsl:apply-templates/> </xsl:copy> </xsl:template> </xsl:stylesheet> 下面分析一下上面counter.xml文件中每条语句的意思: <?xmlversion="1.0"encoding="gb2312"?> 这个就不用说了吧?XML文件都必须包含的部分,XSP是一个XML文件,当然也不能少了它。这里使用GB2312字符集,以便在代码中显示中文字符。 <?cocoon-processtype="xsp"?> 这个处理指令指示cocoon使用XSP来处理此XML文件。 <?cocoon-processtype="xslt"?> <?xml-stylesheethref="counter.xsl"type="text/xsl"?> 这个处理指令指示cocoon使用counter.xsl文件来转换显示此XML文档,对XML文件进行XSLT不是必需的。 <xsp:pagelanguage="java"xmlns:xsp="http://www.apache.org/1999/XSP/Core"> 每个XSP文件都必须包含xsp:page,这是XSP的根元素。同时也必须指定名字空间。而language属性不是必需的,它指定处理逻辑部分使用的语言,默认值为java语言。注意:xsp:page必须包含一个XML文件的根元素,上面代码它包含的根元素是counter。 <xsp:logic></xsp:logic> 这个标签包含的就是XSP的逻辑部分。XSP默认的引入(import)了很多常见的java类(具体是什么看后面的说明),所以你可以直接写java代码,而不必import需要的包和类了。但是如果想引入某个类,可以在xsp:structure中嵌套xsp:include来实现,比如: <xsp:structure> <xsp:include>java.util.Vector</xsp:include> </xsp:structure> 这就相当于importjava.util.Vector <xsp:expr>getCounter()</xsp:expr> 这个标签是求表达式的值,然后显示出来,这是自动做类型转换的。它将此值作为输出XML文件中一个节点的text值存放的。另外,有时候需要在xsp:logic中嵌套着使用xsp:expr,以使xsp:expr输出的内容作为一个节点,这时候你不能直接把xsp:expr放到xsp:logic中,而是将xsp:expr放到xsp:context中,然后再嵌入到xsp:logic中,像下面这样: <xsp:logic> <td> for(inti=0;i<parameterValues.length;i++) { <xsp:content> <xsp:expr>parameterValues[i]</xsp:expr> </xsp:content> <br/> } </td> </xsp:logic> 注意:其中"<"需要转换成<以避免XML解析错误。这通常在写程序时候很麻烦,因为你会经常用到">","<"等等需要转换的符号。你可以将要转换的内容放到<![CDATA]]>标签中,这样XML就会不处理其中的内容,所以上面代码可以这么写: <xsp:logic> <td> <![CDATA[for(inti=0;i<parameterValues.length;i++)]]> { <xsp:content> <xsp:expr>parameterValues[i]</xsp:expr> </xsp:content> <br/> } </td> </xsp:logic> 还要注意的是,一个非空元素不能被xsp:logic截断。比如上面的代码,如果<td>写到xsp:logic外面,就是不正确的: <!--这个td元素是非空的,而且被xsp:logic截断了--> <td> <xsp:logic> <![CDATA[for(inti=0;i<parameterValues.length;i++)]]> { <xsp:content> <xsp:expr>parameterValues[i]</xsp:expr> </xsp:content> <br/> } </xsp:logic> </td> 这只是非常简单的应用,Cocoon附带了很多例子,建议你看看。上面是用java代码在逻辑标签中实现动态内容的,也许你并不熟悉java语言,这并不代表你不能使用XSP。因为XSP还提供了标签库功能,通过使用标签库这样你甚至不懂编程语言也能用标签库来建立XSP页面。关于标签库的使用和XSP其他一些问题下次介绍吧! ---------附------------- 下面这些java类被自动引入: java.io.*; java.util.*; org.w3c.dom.*; org.xml.sax.*; javax.servlet.*; javax.servlet.http.*; org.apache.cocoon.parser.*; org.apache.cocoon.producer.*; org.apache.cocoon.framework.*; org.apache.cocoon.processor.xsp.*;