<p> 一、 XML简介</p><p> XML(可扩展的标注语言)是一种W3C标准,主要用于Web应用程序和服务器之间实现容易的交互、数据的存储与使用。</p><p> 使用XML标准编码的数据具有能容易被人和计算机解释的意义和结构。XML数据是平台和应用程序独立的。不用多说,这本身就使XML成为适合于互联网的一个理想的数据交换格式(事实上,它正是因这一用途而被开发的)。最近,宽带连接的增长及消费者对于越过任何媒体进行数据共享的应用软件的需求意味着,XML Web服务和应用软件正变得越来越丰富。</p><p> XML的发明正是为了解决描述网上丰富的数据的组织问题;而目前为止,这一问题仅能够通过HTML的巧妙使用得到部分地解决。</p><p> 下面是一XML文档的实例:</p><code><?xml version="1.0"?><br /><party><br /> <location>My House</location><br /> <time>7pm</time><br /> <guest><br /><name>John Bloggs</name><br /><item>Crate of Fosters</item><br /> </guest><br /> <guest><br /><name>Sara Bloggs</name><br /><item>Umbrella</item><br /> </guest><br /> <guest><br /><name>David Fig</name><br /><item>Bombay Mix</item><br /> </guest><br /></party></code></p><p> 如果你以前没见过XML,那么你可以认为它看起来象HTML。HTML是一种SGML应用程序,而XML是它的一个子集。然而,其相似性还包括它们具有相似的标注分隔符。</p><p> 仅需看一下上面的XML片断,我们就能看到,该数据是描述一个具有一些客人的聚会;其中,每一个客人相应于一项。用于描述数据的标签名完全由作者来选择。所有XML标准要求:数据必须是一致的并且用于描述数据的标签为良构的。我们可以进一步用一种文档类型声明(DTD)或一个XML模式来强制数据的完整性。然而为简化起见,我们在本文中将仅使用普通的XML。</p>
<p> </p>
<p> 二、 XML应用程序</p><p> 刚才,我们已经看到了如何使用XML来描述任何种类的数据。事实上,XML已经在今天的许多Web应用程序中得到广泛使用,下面是一些著名的应用描述:</p><p> ・ XHTML-这是使用最广泛的XML应用程序之一。它类似基于HTML的SGML-用于描述数据在网页上的显示方式。XHTML使用一DTD来确保所有的文档遵循标准。XHTML的出现使Web程序员的开发稍微容易了一些;然而,一种完全兼容于CSS和XHTML标准的web浏览器尚未出现。</p><p> ・ XML-RPC-远程过程调用(RPC),应用于分布式应用程序中以调用远程计算机上的过程。XML-RPC使用XML对关于过程调用的信息进行编码,并且使用HTTP把它发送到接收计算机。然后,过程的返回值被再次用XML编码并用HTTP连接发送回调用者计算机。</p><p> ・ RSS-真正简单的聚合/丰富的站点摘要,它是一种用来聚合web站点内容(例如新闻、文章、共享价格和链接等)的方法,它用一个特殊的应用程序(一个聚合器)定期更新用户PC上的RSS回馈。该RSS数据是使用XML进行编码和传输的。</p><p> ・ AJAX-异步的JavaScript和XML,允许web开发者创建具有丰富特征的事件驱动的运行在web浏览器上的web应用程序。其中,JavaScript用于把XML编码的数据发送到服务器端脚本(或从服务器端接收XML编码的数据),并允许局部的实时的页面更新而不需要更新所有页面内容。</p><p> 上面仅仅是XML的可能的应用的一部分。在以后文章中,我们将分析如何在PHP中使用这些应用软件。</p><p> 三、 在PHP中使用XML</p><p> 自从PHP 5.0以来,PHP能与XML交互的可用选项显著地增加。而PHP版本4所能提供的是不稳定的而且是非w3c兼容的DOM XML扩展。</p><p> 下面,我将集中讨论PHP 5所提供给我们的三个允许我们与XML交互的方法:DOM,简单XML和XPath。在可能之处,我将建议最适合于每种方法的条件和数据。所有的示例代码将使用XML数据源来描述一个库及其中包含的书。</p>
<p> </p>
<code><xml version="1.0"?><br /><library><br /> <categories><br /><category cid="1">Web Development</category><br /><category cid="2">Database Programming</category><br /><category cid="3">PHP</category><br /><category cid="4">Java</category><br /> </categories><br /> <books><br /> <book><br /><title>Apache 2</title><br /><author>Peter Wainwright</author><br /><publisher>Wrox</publisher><br /><category>1</category><br /> </book><br /> <book><br /><title>Advanced PHP Programming</title><br /><author>George Schlossnagle</author><br /><publisher>Developer Library</publisher><br /><category>1</category><br /><category>3</category><br /> </book><br /> <book><br /><title>Visual FoxPro 6 - Programmers Guide</title><br /><author>Eric Stroo</author><br /><publisher>Microsoft Press</publisher><br /><category>2</category><br /> </book><br /> <book><br /><title>Mastering Java 2</title><br /><author>John Zukowski</author><br /><publisher>Sybex</publisher><br /><category>4</category><br /> </book><br /></books><br /></library></code></p><p> 四、 DOM</p><p> DOM PHP扩展名允许使用W3C DOM API在XML文档上进行操作。在PHP 5出现之前,这是PHP能存取XML文档的唯一方法。如果你在JavaScript中使用了DOM,那么会认识到这些对象模型几乎是一样的。</p>
<p> </p>
<p> 由于DOM方法在遍历和操作XML文档时比较罗嗦,所以任何DOM兼容的代码都有明显的优点-与任何其它实现相同的W3C兼容的对象模型的API兼容。</p><p> 在下面的实例代码中,我们使用DOM来显示关于每本书的信息。首先,我们遍历一下列表目录,把它们的Id和相应的名字装载到一个索引数组中。然后,我们显示每本书的一个简短描述:</p><p> PHP:</p><code><?php<br /> /*这里我们必须指定XML版本:也即是1.0 */<br /> $xml = new DomDocument('1.0');<br /> $xml->load('xml/library.xml');<br /> /*首先,创建一个目录列表*/<br /> $categories = array();<br /> $XMLCategories = $xml->getElementsByTagName('categories')->item(0);<br /> foreach($XMLCategories->getElementsByTagName('category') as $categoryNode) {<br />/*注意我们是如何得到属性的*/<br />$cid = $categoryNode->getAttribute('cid');<br />$categories[$cid] = $categoryNode->firstChild->nodeValue;<br /> }<br />?><br /><html><br /><head><br /><title>XML Library</title><br /></head><br /><body><br /><?<br /> php foreach($xml->getElementsBytagName('book') as $book):<br /> /*查找标题*/<br /> $title = $book->getElementsByTagName('title')->item(0)->firstChild->nodeValue;<br /> /*查找作者-为了简化起见,我们假设仅仅有一个作者*/<br /> $author = $book->getElementsByTagName('author')->item(0)->firstChild->nodeValue;<br /> /* 列表目录*/<br /> $bookCategories = $book->getElementsByTagName('category');<br /> $catList = '';<br /> foreach($bookCategories as $category) {<br />$catList .= $categories[$category->firstChild->nodeValue] . ', ';<br /> }<br /> $catList = substr($catList, 0, -2); ?><br /><div><br /><h2><?php echo($title) ?></h2><br /><p><b>Author:</b>: <?php echo($author) ?></p><br /><p><b>Categories: </b>: <?php echo($catList) ?></p><br /></div><br /><? php endforeach; ?><br /></html></code></p>
<p> </p>
<p> 再提一下,修改XML是较麻烦的。例如,添加一个目录的代码如下:</p><p> PHP:</p><code>function addCategory(DOMDocument $xml, $catID, $catName) {<br /> $catName = $xml->createTextNode($catName); //创建一个结点以存储文本<br /> $category = $xml->createElement('category'); //创建一个目录元素<br /> $category->appendChild($catName); //把文本添加到目录元素上<br /> $category->setAttribute('cid', $catID); //设置目录的ID<br /> $XMLCategories = $xml->getElementsByTagName('categories')->item(0);<br /> $XMLCategories->appendChild($category); //添加新目录<br />}</code></p><p> 五、 保存XML</p><p> 你可以使用save()和saveXML()方法之一来把DOM描述转换回XML字符串描述。save()方法用一指定的命名把XML保存到一个文件中,而saveXML()从文档的部分或整体中返回一个字符串。</p><code>$xml->save('xml/library.xml');<br />//保存全部文件<br />$categories=$xml->saveXML($XMLCategories);<br />//返回一个包含种类的字符串</code></p><p> 为了说明把DOM兼容的代码移植到另外的语言是如何容易,下面是用JavaScript形式实现的与以上功能相同的代码:</p><code>Javascript:<br />function doXML(){<br /> /* 首先创建一个种类列表*/<br /> var categories = Array();<br /> var XMLCategories = xml.getElementsByTagName('categories')[0];<br /> var theCategories = XMLCategories.getElementsByTagName('category');<br /> for (var i = 0; i < theCategories.length; i++) {<br />/* 注意我们是怎样得到属性的*/<br />var cid = theCategories[i].getAttribute('cid');<br />categories[cid] = theCategories[i].firstChild.nodeValue;<br /> }<br /> var theBooks = xml.getElementsByTagName('book');<br /> for(var i = 0; i < theBooks.length; i++) {<br />var book = theBooks[i];<br />/* 查找标题*/<br />var title = book.getElementsByTagName('title')[0].firstChild.nodeValue;<br />/* 查找作者-为简单起见,我们假定仅有一个作者*/<br />var author = book.getElementsByTagName('author')[0].firstChild.nodeValue;<br />/* 列出种类*/<br />var bookCategories = book.getElementsByTagName('category');<br />var catList = '';<br />for(var j = 0; j < bookCategories.length; j++) {<br /> catList += categories[bookCategories[j].firstChild.nodeValue] + ', ';<br />}<br />catList = catList.substring(0, catList.length -2);<br />document.open();<br />document.write("<h2>" + title + "</h2>");<br />document.write("<p><b>Author:</b>: " + author + "</p>");<br />document.write("<p><b>Categories: </b>: " + catList + "</p>");<br /> }<br /> document.close();<br />}</code></p>
<p> </p>
<p> 六、 简单XML</p><p> 简单XML确实简单。它允许使用对象和数组存取方法来存取一个XML文档及其元素和属性。操作方式很简单:</p><p> ・ 元素(Element)-这些被描述为SimpleXMLElement对象的单个属性。当有多个作为文档或元素的子元素存在时,每个元素能被使用数组索引标志加以存取。</p><code>$xml->books;//返回元素"books"<br />$xml->books->book[0];//返回在books元素中的第一本书</code></p><p> ・ 属性(Attribute)-元素的属性是通过关联数组标志来存取和设置的,此时每一个索引对应于一个属性名。</p><code>$category['cid'];//返回cid属性的值</code></p><p> ・ 元素数据(Element Data)-为了检索包含在一个元素内的文本数据,必须使用(string)显式地把它被转换为一个字符串或使用print或echo输出它。如果一个元素包含多个文本结点,那么它们将按被找到的顺序连接起来。</p><code>echo ($xml->books->book[0]->title);//显示第一本书的标题</code></p><p> 下面是使用简单XML进行转换的原来的实例。为了装载XML文件,我们使用simplexml_load_file()函数,由它来分析该XML文件并且把它装载进一个SimpleXMLElement对象中:</p><p> PHP:</p><code><?php<br />$xml = simplexml_load_file('xml/library.xml');<br />/* 把一个列表的目录装载到一个数组中*/<br />$categories = array();<br />foreach($xml->categories->category as $category) {<br /> $categories[(string) $category['cid']] = (string) $category;<br />}<br />?><br /><html><br /><head><br /><title>XML Library</title><br /></head><br /><body><br /><?php foreach($xml->books->book as $book):<br />/* 列举目录*/<br />$catList = '';<br />foreach($book->category as $category) {<br /> $catList .= $categories[((string) $category)] . ', ';<br />}<br />$catList = substr($catList, 0, -2); ?><br /><div><br /><h2><?php echo($book->title) ?></h2><br /><p><b>Author:</b>: <?php echo($book->author) ?></p><br /><p><b>Categories: </b>: <? php echo($catList) ?></p><br /></div><br /><? php endforeach; ?><br /></html></code></p>
<p> </p>
<p> 七、 修改XML</p><p> 尽管文本数据和属性值可以通过使用简单XML加以设置,但是不能新建这些对象。然而,SimpleXM的确提供了一种方法来实现DomElement对象和DomElement对象之间的转换。为此,我修改了addCategory()函数来说明如何使用simplexml_import_dom()函数以添加目录和把该文档转换回简单的XML格式:</p><p> PHP:</p><code>function addCategory(SimpleXMLElement &$sXML, $catID, $catName) {<br /> $xml = new DOMDocument;<br /> $xml->loadXML($sXML->asXML());<br /> $catName = $xml->createTextNode($catName); //创建一个结点来存放该文本<br /> $category = $xml->createElement('category'); //创建一个目录元素<br /> $category->appendChild($catName); //把文本添加到目录元素<br /> $category->setAttribute('cid', $catID); //设置目录id<br /> $XMLCategories = $xml->getElementsByTagName('categories')->item(0);<br /> $XMLCategories->appendChild($category); //添加新目录<br /> $sXML = simplexml_import_dom($xml);<br /> return $sXML;<br />}</code></p><p> 同样,SimpleXMLElement对象的asXML()函数可以用来检索XML字符串并把它保存回一个文件中。</p><p> 八、 xPath</p><p> 毫无疑问,Xpath是"XML蛋糕之上的樱桃"。XPath允许你使用象SQL一样的查询来查找一个XML文档中的特定信息。DOM和SimpleXML都有内置的对XPath的支持,如SQL,可以被用来提取你想从一XML文档中提取的任何内容。</p><p> ・ //category-查找所有的在文档中出现的任何category。</p><p> ・ /library/books-查找所有作为library的孩子出现的books</p>
<p> </p>
<p> ・ /library/categories/category[@cid]-查找所有作为library/categories的孩子出现且属性为cid的category。</p><p> ・ /library/categories/category[@att='2']-查找所有作为library/categories的孩子且具有属性cid的值为2出现的category。</p><p> ・ /library/books/book[title='Apache 2']-查找所有作为/library/books的孩子且其标题元素有一个值为Apache 2出现的book。</p><p> 其实,这仅是xPath冰山之一角。你可以使用xPath来创建大量复杂的查询以便从你的文档中提取几乎任何信息。我再次修改了示例代码来向你展示使用xPath是多么轻松愉快的事情。</p><p> PHP:</p><code><?php<br />$xml = simplexml_load_file('xml/library.xml');<br />?><br /><html><br /><head><br /><title>XML Library</title><br /></head><br /><body><br /><?php foreach(((array)$xml->xpath("/library/books/book")) as $book):<br />/*列表目录*/<br />$catList = '';<br />foreach($book->category as $category) {<br /> /*得到具有这个ID的目录*/<br /> $category = $xml->xpath("/library/categories/category[@cid='$category']");<br /> $catList .= (string) $category[0] . ', ';<br />}<br />$catList = substr($catList, 0, -2); ?><br /><div><br /><h2><?php echo($book->title) ?></h2><br /><p><b>Author:</b>: <?php echo($book->author) ?></p><br /><p><b>Categories: </b>: <?php echo($catList) ?></p><br /></div><br /><?php endforeach; ?><br /></html></code></p></p><p> 九、 DOM和XPath</p><p> 在DOM中计算XPath查询需要创建一个DOMXPath对象,下面的evaluate()函数返回一个DOMElement数组。</p><code>$xPath = new DOMXPath($xml);<br />$xPath->evaluate("/library/books/book[title='Apache 2']");</code></p><p> 十、 结论</p><p> 现在,我们学习了如何使用了PHP提供给我们的工具来与XML交互。至此,我们已经被"武装起来"并准备好深入钻研XML应用程序了。在下一篇文章中,我们将讨论AJAX及其如何应用于象Google这样的站点开发的。</p></p>