使用PHP DOMDocument建立動態XML檔案【建議閱讀:php入門教學】
當處理基於XML應用程式時,開發者經常需要建立XML編碼資料結構。例如,Web中基於使用者輸入的XML狀態模板,伺服器請求XML語句,以及基於運行時間參數的客戶回應。
儘管XML資料結構的建構比較費時,但如果使用成熟的PHP DOM應用程式接口,一切都會變得簡單明了。本文將向你介紹PHP DOM應用程式介面的主要功能,示範如何產生一個正確的XML完整檔案並將其儲存到磁碟中。
建立文件類型聲明
一般而言,XML聲明放在文件頂部。在PHP中聲明十分簡單:只需實例化一個DOM文檔類別的物件並賦予它一個版本號。檢視程序清單A:
程序清單 A
<?php // create doctype $dom = new DOMDocument("1.0"); // display document in browser as plain text // display document in browser as plain text // for readability purposes header("Content-Type: text/plain"); // save and display tree echo $dom->saveXML(); ?>
請注意DOM文件物件的saveXML()方法。稍後我再詳細介紹這個方法,現在你只需要簡單地認識到它用於輸出XML文件的當前快照到一個文件或瀏覽器。在本例,為增強可讀性,我已經將ASCII碼文字直接輸出至瀏覽器。在實際應用中,可將以text/XML頭檔傳送至瀏覽器。
如在瀏覽器中查看輸出,可看到以下程式碼:
新增元素和文字節點
XML真正強大的功能是來自其元素與封裝的內容。幸運的是,一旦你初始化DOM文檔,很多操作變得很簡單。此過程包含以下兩步驟:
對想要新增的每一元素或文字節點,透過元素名稱或文字內容呼叫DOM文件物件的createElement()或createTextNode()方法。這將建立對應於元素或文字節點的新物件。
透過呼叫節點的appendChild()方法,並將其傳遞給上一個步驟中建立的對象,並在XML文件樹中將元素或文字節點新增至父節點。
以下範例將清楚地示範這2步驟,請查看程序清單B。
程式清單 B
<?php // create doctype $dom = new DOMDocument("1.0"); // display document in browser as plain text // for readability purposes header("Content-Type: text/plain"); // create root element $root = $dom->createElement("toppings"); $dom->appendChild($root); // create child element $item = $dom->createElement("item"); $root->appendChild($item); // create text node $text = $dom->createTextNode("pepperoni"); $item->appendChild($text); // save and display tree echo $dom->saveXML(); ?>
這 裡,我先建立一個名字為
<?xml version="1.0"?> <toppings> <item>pepperoni</item> </toppings>
如果你想再增加一個topping,只需建立另一個
程式清單C
<?php // create doctype $dom = new DOMDocument("1.0"); // display document in browser as plain text // for readability purposes header("Content-Type: text/plain"); // create root element $root = $dom->createElement("toppings"); $dom->appendChild($root); // create child element $item = $dom->createElement("item"); $root->appendChild($item); // create text node $text = $dom->createTextNode("pepperoni"); $item->appendChild($text); // create child element $item = $dom->createElement("item"); $root->appendChild($item); // create another text node $text = $dom->createTextNode("tomato"); $item->appendChild($text); // save and display tree echo $dom->saveXML(); ?>
以下是執行程式清單C後的輸出:
<?xml version="1.0"?> <toppings> <item>pepperoni</item> <item>tomato</item> </toppings>
新增屬性
透過使用屬性,你也可以添加適合的資訊到元素。對於PHP DOM API,新增屬性需要兩步驟:先用DOM文件物件的createAttribute()方法建立擁有此屬性名字的節點,然後將文件節點新增至擁有屬性值的屬性節點。詳見程序清單D。
程式清單D
<?php // create doctype $dom = new DOMDocument("1.0"); // display document in browser as plain text // for readability purposes header("Content-Type: text/plain"); // create root element $root = $dom->createElement("toppings"); $dom->appendChild($root); // create child element $item = $dom->createElement("item"); $root->appendChild($item); // create text node $text = $dom->createTextNode("pepperoni"); $item->appendChild($text); // create attribute node $price = $dom->createAttribute("price"); $item->appendChild($price); // create attribute value node $priceValue = $dom->createTextNode("4"); $price->appendChild($priceValue); // save and display tree echo $dom->saveXML(); ?>
輸出如下:
<?xml version="1.0"?> <toppings> <item price="4">pepperoni</item> </toppings>
新增CDATA模組和製程精靈
雖然不常使用CDATA模組和流程精靈,但透過呼叫DOM文件物件的createCDATASection()和createProcessingInstruction()方法, PHP API 也能很好地支援CDATA和流程嚮導,請見程式清單E。
程式清單E
<?php // create doctype // create doctype $dom = new DOMDocument("1.0"); // display document in browser as plain text // for readability purposes header("Content-Type: text/plain"); // create root element $root = $dom->createElement("toppings"); $dom->appendChild($root); // create child element $item = $dom->createElement("item"); $root->appendChild($item); // create text node $text = $dom->createTextNode("pepperoni"); $item->appendChild($text); // create attribute node $price = $dom->createAttribute("price"); $item->appendChild($price); // create attribute value node $priceValue = $dom->createTextNode("4"); $price->appendChild($priceValue); // create CDATA section $cdata = $dom->createCDATASection(" Customer requests that pizza be sliced into 16 square pieces "); $root->appendChild($cdata); // create PI $pi = $dom->createProcessingInstruction("pizza", "bake()"); $root->appendChild($pi); // save and display tree echo $dom->saveXML(); ?>
輸出如下:
<?xml version="1.0"?> <toppings> <item price="4">pepperoni</item> <![CDATA[ Customer requests that pizza be sliced into 16 square pieces ]]> <?pizza bake()?> </toppings>
儲存結果
一旦已經實現你的目標,就可以將結果保存在一個檔案或儲存於PHP的變數。透過呼叫帶有檔案名稱的save()方法可以將結果保存在檔案中,而透過呼叫saveXML()方法可儲存於PHP的變數。請參考以下實例(程式清單F):
程式清單 F
<?php // create doctype $dom = new DOMDocument("1.0"); // create root element $root = $dom->createElement("toppings"); $dom->appendChild($root); $dom->formatOutput=true; // create child element $item = $dom->createElement("item"); $root->appendChild($item); // create text node $text = $dom->createTextNode("pepperoni"); $item->appendChild($text); // create attribute node $price = $dom->createAttribute("price"); $item->appendChild($price); // create attribute value node $priceValue = $dom->createTextNode("4"); $price->appendChild($priceValue); // create CDATA section $cdata = $dom->createCDATASection(" Customer requests that pizza be sliced into 16 square pieces "); $root->appendChild($cdata); // create PI $pi = $dom->createProcessingInstruction("pizza", "bake()"); $root->appendChild($pi); // save tree to file $dom->save("order.xml"); // save tree to string $order = $dom->save("order.xml"); ?>
以下是實際的例子,大家可以測試下。
xml.php(生成xml) <? $conn = mysql_connect('localhost', 'root', '123456') or die('Could not connect: ' . mysql_error()); mysql_select_db('vdigital', $conn) or die ('Can\'t use database : ' . mysql_error()); $str = "SELECT id,username FROM `admin` GROUP BY `id` ORDER BY `id` ASC"; $result = mysql_query($str) or die("Invalid query: " . mysql_error()); if($result) { $xmlDoc = new DOMDocument(); if(!file_exists("01.xml")){ $xmlstr = "<?xml version='1.0' encoding='utf-8' ?><message></message>"; $xmlDoc->loadXML($xmlstr); $xmlDoc->save("01.xml"); } else { $xmlDoc->load("01.xml");} $Root = $xmlDoc->documentElement; while ($arr = mysql_fetch_array($result)){ $node1 = $xmlDoc->createElement("id"); $text = $xmlDoc->createTextNode(iconv("GB2312","UTF-8",$arr["id"])); $node1->appendChild($text); $node2 = $xmlDoc->createElement("name"); $text2 = $xmlDoc->createTextNode(iconv("GB2312","UTF-8",$arr["username"])); $node2->appendChild($text2); $Root->appendChild($node1); $Root->appendChild($node2); $xmlDoc->save("01.xml"); } } mysql_close($conn); ?>
test.php(應用測試)
<? $xmlDoc = new DOMDocument(); $xmlDoc->load("http://localhost/xml/xml.php"); $x=$xmlDoc->getElementsByTagName('name'); for ($i=0; $i<=$x->length-1; $i++) { if(strpos($x->item($i)->nodeValue,"fang")!==false) { echo $x->item($i)->parentNode->childNodes->item(1)->nodeValue; } } ?>