使用PHP和XSL stylesheets轉(zhuǎn)換XML文檔
PHP是不少在Web開(kāi)發(fā)領(lǐng)域奮戰(zhàn)的勇士們所選用的武器,因?yàn)樗且环N很直觀的編程語(yǔ)言,有強(qiáng)大的函數(shù),良好的跨平臺(tái)兼容性,還有它是免費(fèi)的。從網(wǎng)上的小商店到大型企業(yè)的網(wǎng)站都能看到PHP的影子。
PHP有一點(diǎn)特性經(jīng)常被人們忽視,那就是和XSL stylesheets合作對(duì)XML進(jìn)行解析的能力。下面就讓我們來(lái)看看怎樣在PHP中設(shè)置一個(gè)XSL解析器以及你該如何使用這一功能。
例子列表A是一個(gè)簡(jiǎn)單的訂單文檔,我們會(huì)將這個(gè)文檔輸入XSL解析器。同時(shí),列表B中的XSL stylesheet也會(huì)被輸入XSL解析器。
Listing A: order.xml
<?xml version='1.0' ?><Order> <Account>9900234</Account> <Item id='1'> <SKU>1234</SKU> <PricePer>5.95</PricePer> <Quantity>100</Quantity> <Subtotal>595.00</Subtotal> <Description>Super Widget Clamp</Description> </Item> <Item id='2'> <SKU>6234</SKU> <PricePer>22.00</PricePer> <Quantity>10</Quantity> <Subtotal>220.00</Subtotal> <Description>Mighty Foobar Flange</Description> </Item> <Item id='3'> <SKU>9982</SKU> <PricePer>2.50</PricePer> <Quantity>1000</Quantity> <Subtotal>2500.00</Subtotal> <Description>Deluxe Doohickie</Description> </Item> <Item id='4'> <SKU>3256</SKU> <PricePer>389.00</PricePer> <Quantity>1</Quantity> <Subtotal>389.00</Subtotal> <Description>Muckalucket Bucket</Description> </Item> <NumberItems>1111</NumberItems> <Total>3704.00</Total> <OrderDate>07/07/2002</OrderDate> <OrderNumber>8876</OrderNumber></Order> Listing B: order.xsl
<?xml version='1.0'; ?><xsl:stylesheet version='1.0' xmlns:xsl='http://www.w3.org/1999/XSL/Transform'> <xsl:param name='column' select=''SKU''/><xsl:param name='order' select=''ascending''/> <xsl:template match='/'> <html> <body> <xsl:apply-templates select='Order'> <xsl:with-param name='sortcolumn' select='$column' /> <xsl:with-param name='sortorder' select='$order' /> </xsl:apply-templates> </body> </html> </xsl:template>
<xsl:template match='Order'> <xsl:param name='sortcolumn' /> <xsl:param name='sortorder' /> <table border='1'> <tr> <th>Account</th> <th>SKU</th> <th>Description</th> <th>Price</th> <th>Quantity</th> <th>Subtotal</th> </tr> <xsl:apply-templates select='Item'> <xsl:sort select='*[name()=$sortcolumn]'; order='{$sortorder}' /> </xsl:apply-templates> </table> </xsl:template>
<xsl:template match='Item'> <tr> <td><xsl:value-of select='../Account' /></td> <td><xsl:value-of select='SKU' /></td> <td><xsl:value-of select='Description' /></td> <td><xsl:value-of select='PricePer' /></td> <td><xsl:value-of select='Quantity' /></td> <td><xsl:value-of select='Subtotal' /></td> </tr> </xsl:template>;</xsl:stylesheet> 概述在這個(gè)例子中我們主要用到PHP中的三個(gè)XSL函數(shù)。首先我們要?jiǎng)?chuàng)建一個(gè)XSL引擎的實(shí)例,然后把所有要輸入的文檔輸入這個(gè)XSL引擎進(jìn)行處理,并得到返回結(jié)果,最后,當(dāng)我們?cè)僖膊恍枰@個(gè)XSL引擎時(shí)就關(guān)閉它。
創(chuàng)建、處理、關(guān)閉我們將要在內(nèi)存中新建一個(gè)XSL進(jìn)程。為了方便在其他XSL函數(shù)中使用這個(gè)XSL進(jìn)程,PHP會(huì)給我們提供這個(gè)XSL進(jìn)程的句柄,而不是一個(gè)對(duì)象。建立這個(gè)XSL引擎的命令是xslt_create。函數(shù)返回一個(gè)句柄,如下所示:
$handle = xslt_create();
為了真正的解析XML文檔并使XSLT能夠進(jìn)行處理,你必須使用PHP中的xslt_process函數(shù)。這個(gè)函數(shù)需要獲取幾個(gè)不同的參數(shù)。
在這里我們使用一個(gè)很基本的方法,為xslt_process提供三個(gè)參數(shù)。第一個(gè)參數(shù)是我們較早前創(chuàng)建的那個(gè)XSL引擎的句柄。第二個(gè)參數(shù)是輸入的XML文檔的文件名。第三個(gè)參數(shù)是輸入的XSL文件的文件名。這個(gè)函數(shù)會(huì)返回處理結(jié)果。下面是例子:
$return = xslt_process($handle, $xmlfile, $xslfile);
最后我們要用到的函數(shù)是xslt_free。這個(gè)函數(shù)用來(lái)殺掉內(nèi)存中的XSL引擎實(shí)例并釋放出內(nèi)存空間。它只需要一個(gè)參數(shù),就是內(nèi)存中這個(gè)XSL實(shí)例的句柄。下面是個(gè)例子:
xslt_free($handle);
綜合實(shí)現(xiàn)
下面讓我們結(jié)合上面的各個(gè)代碼片斷實(shí)現(xiàn)PHP通過(guò)XSL stylesheets來(lái)處理XML文檔的方法。我們使用列表A作為我們的輸入XML文檔,列表B作為我們XSL輸入。列表C是這個(gè)例子的完整PHP代碼:
Listing C: order.php
<?php$xmlfile = 'order.xml';$xslfile = 'order.xsl';$args = array('column'=>'Quantity', 'order'=>'descending');$engine = xslt_create();$output = xslt_process($engine, $xmlfile, $xslfile, NULL, NULL, $args);print $output;xslt_free($engine);?>
這里需要注意一點(diǎn),我們?cè)诖a中做了一點(diǎn)變動(dòng)。在XSL stylesheet中,通過(guò)指定一些參數(shù),我們可以改變一些區(qū)域,比如地址。這時(shí)我們要指定訂單上的項(xiàng)目應(yīng)該按數(shù)量遞減方式排列。我們使用PHP的數(shù)組來(lái)存儲(chǔ)名字對(duì)應(yīng)我們的參數(shù),然后通過(guò)xslt_process函數(shù)將名字傳遞給XSL引擎。
本文作者Brian Schaffner是富士通咨詢(xún)公司的副主任。他為富士通的技術(shù)咨詢(xún)公司提供架構(gòu)、設(shè)計(jì)和開(kāi)發(fā)支持。
