組一樣。所有的類變量(當你用PHP3.0時還有方法名)都以字符串的方式輸出。
當發現一個元素時,我們需要增加其相應的記數器來跟蹤它在文檔中出現多少次。在相應的$elements項中的記數元素也要加一。
我們同樣要讓父元素知道目前的元素是它的子元素。因此,目前元素的名稱將會加入到父元素的$childs數組的項目中。最後,目前元素應該記住誰是它的父元素。所以,父元素被加入到目前元素$parents數組的項目中。
顯示統計信息
剩下的代碼在$elements數組和其子數組中循環顯示其統計結果。這就是最簡單的嵌套循環,盡管輸出正確的結果,但代碼既不簡潔又沒有任何特別的技巧,它僅僅是一個你可能每天用他來完成工作的循環。
腳本范例被設計為通過PHP的CGI方式的命令行來調用。因此,統計結果輸出的格式為文本格式。如果你要將腳本運用到互聯網上,那麼你需要修改輸出函數來產生Html格式。
總結
Exapt是PHP的XML解析器。作為基於事件的解析器,它不產生文檔的結構描述。但通過提供底層訪問,這就使得可以更好地利用資源和更快地訪問。
作為一個不檢查有效性的解析器,Expat忽略與XML文檔連接的DTD,但如果文檔的格式不完整,它將會隨著出錯信息而停止。
提供事件處理函數來處理文檔
建立自己的事件結構例如棧和樹來獲得XML結構信息標記的優點。
每天都有新的XML程序出現,而PHP對XML的支持也不斷加強(例如,增加了支持基於DOM的XML解析器LibXML)。
有了PHP和Expat,你就可以為即將出現的有效、開放和獨立於平台的標准作准備。
范例
<?
/*****************************************************************************
* 名稱:XML解析范例:XML文檔信息統計
* 描述
* 本范例通過PHP的Expat解析器收集和統計XML文檔的信息(例如:每個元素出現的次數、父元素和子元素
* XML文件作為一個參數 ./XMLstats_PHP4.PHP3 test.XML
* $Requires: Expat 要求:Expat PHP4.0編譯為CGI模式
*****************************************************************************/
// 第一個參數是XML文件
$file = $argv[1];
// 變量的初始化
$elements = $stack = array();
$total_elements = $total_chars = 0;
// 元素的基本類
class element
{
var $count = 0;
var $chars = 0;
var $parents = array();
var $childs = array();
}
// 解析XML文件的函數
function XML_parse_from_file($parser, $file)
{
if(!file_exists($file))
{
dIE("Can@#t find file \"$file\".");
}
if(!($fp = @fopen($file, "r")))
{
dIE("Can@#t open file \"$file\".");
}
while($data = fread($fp, 4096))
{
if(!XML_parse($parser, $data, feof($fp)))
{
return(false);
}
}
fclose($fp);
return(true);
}
// 輸出結果函數(方框形式)
function print_box($title, $value)
{
printf("\n+%@#-60s+\n", "");
printf("|%20s", "$title:");
printf("%14s", $value);
printf("%26s|\n", "");
printf("+%@#-60s+\n", "");
}
// 輸出結果函數(行形式)
function print_line($title, $value)
{
printf("%20s", "$title:");
printf("%15s\n", $value);
}
// 排序函數
function my_sort($a, $b)
{
return(is_object($a) && is_object($b) ? $b->count - $a->count: 0);
}
function start_element($parser, $name, $attrs)
{
global $elements, $stack;
// 元素是否已在全局$elements數組中?
if(!isset($elements[$name]))
{
// 否-增加一個元素的類實例
$element = new element;
$elements[$name] = $element;
}
// 該元素的記數器加一
$elements[$name]->count++;
// 是否有父元素?
if(isset($stack[count($stack)-1]))
{
// 是-將父元素賦給$last_element
$last_element = $stack[count($stack)-1];
// 如果目前元素的父元素數組為空,初始化為0
if(!isset($elements[$name]->parents[$last_element]))
{
$elements[$name]->parents[$last_element] = 0;
}
// 該元素的父元素記數器加一
$elements[$name]->parents[$last_element]++;
// 如果目前元素的父元素的子元素數組為空,初始化為0
if(!isset($elements[$last_element]->childs[$name]))
{
$elements[$last_element]->childs[$name] = 0;
}
// 該元素的父元素的子元素記數器加一
$elements[$last_element]->childs[$name]++;
}
// 將目前的元素加入到棧中
array_push($stack, $name);
}
function stop_element($parser, $name)
{
global $stack;
// 從棧中將最頂部的元素移去
array_pop($stack);
}
function char_data($parser, $data)
{
global $elements, $stack, $depth;
// 增加目前元素的字符數目
$elements[$stack][count($stack)-1]]->chars += strlen(trim($data));
}
// 產生解析器的實例
$parser = XML_parser_create();
// 設置處理函數
XML_set_element_handler($parser, "start_element", "stop_ele ment");
XML_set_character_data_handler($parser, "char_data");
xml_parser_set_option($parser, XML_OPTION_CASE_FOLDING, 0);
// 解析文件
$ret = XML_parse_from_file($parser, $file);
if(!$ret)
{
dIE(sprintf("XML error: %s at line %d",
xml_error_string(XML_get_error_code($parser)),
XML_get_current_line_number($parser)));
}
// 釋放解析器
XML_parser_free($parser);
// 釋放協助元素
unset($elements["current_element"]);
unset($elements["last_element"]);
// 根據元素的次數排序
uasort($elements, "my_sort");
// 在$elements中循環收集元素信息
while(list($name, $element) = each($elements))
{
print_box("Element name", $name);
print_line("Element count", $element->count);
print_line("Character count", $element->chars);
printf("\n%20s\n", "* Parent elements");
// 在該元素的父中循環,輸出結果
while(list($key, $value) = each($element->parents))
{
print_line($key, $value);
}
if(count($element->parents) == 0)
{
printf("%35s\n", "[root element]");
}
// 在該元素的子中循環,輸出結果
printf("\n%20s\n", "* Child elements");
while(list($key, $value) = each($element->childs))
{
print_line($key, $value);
}
if(count($element->childs) == 0)
{
printf("%35s\n", "[no childs]");
}
$total_elements += $element->count;
$total_chars += $element->chars;
}
// 最終結果
print_box("Total elements", $total_elements);
print_box("Total characters", $total_chars);
?>