分类:
2008-05-29 09:06:36
首先我得承认我喜欢计算机标准。如果每个人都遵从这个行业的标准,互联网将会是一个更好的媒体。使用标准化的数据交换格式才能使开放的和独立于平台的计算模式切实可行。这就是我作为XML爱好者的原因。 php程序员之家 幸运的是,我最喜爱的脚本语言不但支持XML而且对其支持正不断加强。PHP可以让我迅速将XML文档发布到互联网上,收集XML文档的统计信息,将XML文档转换成其它格式。例如,我时常用PHP的XML处理能力来管理我用XML所写的文章和书。 php程序员站 本文中,我将讨论任何用PHP内建的Expat解析器来处理XML文档。通过范例,我将演示Expat的处理方法。同时,范例可以告诉你如何: phperz.com
建立你自己的处理函数 介绍Expat php程序员之家
XML的解析器,同样称为XML处理器,可以使程序访问XML文档的结构和内容。Expat是PHP脚本语言的XML解析器。它同时也运用在其它项目中,例如Mozilla、Apache和Perl。 www~phperz~com 什么是基于事件的解析器? XML解析器的两种基本类型: php程序员之家 基于树型的解析器:将XML文档转换成树型结构。这类解析器分析整篇文章,同时提供一个API来访问所产生树的每个元素。其通用的标准为DOM(文档对象模式)。
基于事件的解析器将报告为三个事件: www~phperz~com 开始元素:greeting phperz.com Expat就是这样的一种基于事件的解析器。当然如果使用Expat,必要时它一样可以在PHP中生成完全的原生树结构。 www~phperz~com 上面Hello-World的范例包括完整的XML格式。但它是无效的,因为既没有DTD(文档类型定义)与其联系,也没有内嵌DTD。 对于Expat,这并没有区别:Expat是一个不检查有效性的解析器,因此忽略任何与文档联系的DTD。但应注意的是文档仍然需要完整的格式,否则Expat(和其他符合XML标准的解析器一样)将会随着出错信息而停止。 phperz.com
作为不检查有效性的解析器,Exapt的快速性和轻巧性使其十分适合互联网程序。 phperz~com 编译Expat php程序员之家
Expat可以编译进PHP3.0.6版本(或以上)中。从Apache1.3.9开始,Expat已经作为Apache的一部分。在Unix系统中,通过-with-xml选项配置PHP,你可以将其编译入PHP。 phperz~com
如果你将PHP编译为Apache的模块,而Expat将默认作为Apache的一部分。在Windows中,你则必须要加载XML动态连接库。 phperz.com XML范例:XMLstats phperz~com 了解Expat的函数的一个办法就是通过范例。我们所要讨论的范例是使用Expat来收集XML文档的统计数据。 对于文档中每个元素,以下信息都将被输出: 该元素在文档中使用的次数 php程序员站
准备 phperz.com 用 于产生XML解析器实例的函数为xml_parser_create()。该实例将用于以后的所有函数。这个思路非常类似于PHP中MySQL函数的连接 标记。在解析文档前,基于事件的解析器通常要求你注册回调函数-用于特定的事件发生时调用。Expat没有例外事件,它定义了如下七个可能事件: php程序员之家 对象 XML解析函数 描述 phperz.com
元素 xml_set_element_handler() 元素的开始和结束 phperz.com
字符数据 xml_set_character_data_handler() 字符数据的开始 phperz.com 外部实体 xml_set_external_entity_ref_handler() 外部实体出现 php程序员站
未解析外部实体 xml_set_unparsed_entity_decl_handler() 未解析的外部实体出现 处理指令 xml_set_processing_instruction_handler() 处理指令的出现 phperz.com 记法声明 xml_set_notation_decl_handler() 记法声明的出现 phperz~com
默认 xml_set_default_handler() 其它没有指定处理函数的事件 www~phperz~com
所有的回调函数必须将解析器的实例作为其第一个参数(此外还有其它参数)。 php程序员站 对于本文最后的范例脚本。你需要注意的是它既用到了元素处理函数又用到了字符数据处理函数。元素的回调处理函数通过xml_set_element_handler()来注册。 phperz.com
这个函数需要三个参数: 解析器的实例 php程序员站 例如,Expat将三个参数传递给开始元素的处理函数。在脚本范例中,其定义如下: phperz.com
function start_element($parser, $name, $attrs) php程序员站 第一个参数是解析器标示,第二个参数是开始元素的名称,第三参数为包含元素所有属性和值的数组。 php程序员之家
一旦你开始解析XML文档,Expat在遇到开始元素是都将调用你的start_element()函数并将参数传递过去。 XML的Case Folding选项 用xml_parser_set_option ()函数将Case folding选项关闭。这个选项默认是打开的,使得传递给处理函数的元素名自动转换为大写。但XML对大小写是敏感的(所以大小写对统计XML文档是非 常重要的)。对于我们的范例,case folding选项必须关闭。 解析文档 在完成所有的准备工作后,现在脚本终于可以解析XML文档: Xml_parse_from_file(),一个自定义的函数,打开参数中指定的文件,并以4kb的大小进行解析 phperz.com
当解析文档时,对于Expat需要强调问题的是:如何保持文档结构的基本描述? 如前所述,基于事件的解析器本身并不产生任何结构信息。 phperz.com 不 过标签(tag)结构是XML的重要特性。例如,元素序列 为了产生文档结构的镜像,脚本至少需要知道目前元素的父元素。用Exapt的API是无法实现的,它只报告目前元素的事件,而没有任何前后关系的信息。因此,你需要建立自己的栈结构。 php程序员站 脚本范例使用先进后出(FILO)的栈结构。通过一个数组,栈将保存全部的开始元素。对于开始元素处理函数,目前的元素将被array_push()函数推到栈的顶部。相应的,结束元素处理函数通过array_pop()将最顶的元素移走。 对于序列 开始元素book:将"book"赋给栈的第一个元素($stack[0])。 phperz.com
收集数据 为了收集每个元素的信息,脚本需要记住每个元素的事件。通过使用一个全局的数组变量$elements来保存文档中所有不同的元素。数组的项目是元素类的实例,有4个属性(类的变量) $count -该元素在文档中被发现的次数 注意:PHP的一个特性是你可以通过while(list() = each())loop遍历整个类结构,如同你遍历整个相应的数组一样。所有的类变量(当你用PHP3.0时还有方法名)都以字符串的方式输出。 www~phperz~com 当发现一个元素时,我们需要增加其相应的记数器来跟踪它在文档中出现多少次。在相应的$elements项中的记数元素也要加一。 www~phperz~com 我们同样要让父元素知道目前的元素是它的子元素。因此,目前元素的名称将会加入到父元素的$childs数组的项目中。最后,目前元素应该记住谁是它的父元素。所以,父元素被加入到目前元素$parents数组的项目中。 显示统计信息 php程序员站
剩下的代码在$elements数组和其子数组中循环显示其统计结果。这就是最简单的嵌套循环,尽管输出正确的结果,但代码既不简洁又没有任何特别的技巧,它仅仅是一个你可能每天用他来完成工作的循环。 www~phperz~com
脚本范例被设计为通过PHP的CGI方式的命令行来调用。因此,统计结果输出的格式为文本格式。如果你要将脚本运用到互联网上,那么你需要修改输出函数来产生HTML格式。 php程序员站 总结 phperz.com
Exapt是PHP的XML解析器。作为基于事件的解析器,它不产生文档的结构描述。但通过提供底层访问,这就使得可以更好地利用资源和更快地访问。 作为一个不检查有效性的解析器,Expat忽略与XML文档连接的DTD,但如果文档的格式不完整,它将会随着出错信息而停止。 提供事件处理函数来处理文档 php程序员站 有了PHP和Expat,你就可以为即将出现的有效、开放和独立于平台的标准作准备。 www~phperz~com 范例 // 第一个参数是XML文件 // 变量的初始化 // 元素的基本类 php程序员站 // 解析XML文件的函数 phperz.com if(!($fp = @fopen($file, "r"))) while($data = fread($fp, 4096)) fclose($fp); phperz~com return(true); // 输出结果函数(方框形式) phperz.com // 输出结果函数(行形式) php程序员之家 // 排序函数 function start_element($parser, $name, $attrs) phperz~com // 元素是否已在全局$elements数组中? www~phperz~com // 该元素的记数器加一 phperz~com // 是否有父元素? // 如果目前元素的父元素数组为空,初始化为0 // 该元素的父元素记数器加一 phperz.com // 如果目前元素的父元素的子元素数组为空,初始化为0 phperz~com if(!isset($elements[$last_element]->childs[$name])) phperz~com // 该元素的父元素的子元素记数器加一 phperz~com // 将目前的元素加入到栈中 function stop_element($parser, $name) php程序员站 // 从栈中将最顶部的元素移去 function char_data($parser, $data) // 增加目前元素的字符数目 php程序员站
// 产生解析器的实例 phperz~com // 设置处理函数 // 解析文件 // 释放解析器 php程序员站 // 释放协助元素 // 根据元素的次数排序 php程序员站
// 在$elements中循环收集元素信息 php程序员站
print_line("Element count", $element->count); phperz.com
printf("n%20sn", "* Parent elements"); php程序员之家 // 在该元素的父中循环,输出结果 // 在该元素的子中循环,输出结果 www~phperz~com $total_elements += $element->count; // 最终结果 phperz.com
|