使用PHP和AJAX的XML编程

  在SOA架构中,程序员经常使用XML在应用程序之间交换结构化和半结构化的数据。XML及其相关技术文档对象模型(DocumentObjectModel,DOM)、XPath、HTTP、XQuery和可扩展样式表语言转换(ExtensibleStylesheetLanguageTransformations,XSLT)为快速应用程序开发提供了一个强大的环境。构建在这些技术之上的应用程序将占用更小的内存空间,需要更低的维护成本,同时又拥有更高的品质和灵活性。

  DB2和其他关系数据库的XML方面的特性已经相当成熟,因此除了存储和管理关系数据之外,它们还是存储和管理XML数据的理想选择。DB29XML支持(称为pureXML)提供了以纯XML的形式(换句话说,就是带注释的、树型的分层存储)存储XML的能力。在DB29中,XML数据可以用XML模式索引,可以从关系数据组合而成,可以分解为关系数据,可以查询和转换,可以独立发布,或者通过混合使用SQL/XML和XQuery与关系数据组合起来。

  Web浏览器也正在为客户机脚本提供更多的功能来有效地处理XML。通过使用异步JavaScript和XML(AsynchronousJavaScriptandXML,Ajax),Web页面现在可以直接对应用服务器进行远程过程调用,并且可以在任何返回的XML数据上使用DOMAPI。

  本文将展示如何利用DB2XML、Ajax和PHPHypertextPreprocessor(PHP)提供的功能来编写简单的基于XML的应用程序。通过示例场景的帮助,您将学习如何在JavaScript中调用PHP应用程序;如何使用DOM和SimpleXMLAPI修改XML数据;如何将XML从客户机传送到应用程序再到数据库;以及如何创建PHPWeb服务来使用SQL/XML和XQuery发布关于XML数据的报告。

  大多数应用程序都用于创建、存储、操纵和呈现业务数据。对象包装是指将业务数据绑在一起,使业务逻辑更容易处理它们。这些包装器对象的很多功能都是根据关系和格式化规则来提供业务数据的结构,并使业务逻辑能够操纵、发布和串行化封装的数据。

  图1阐释了使用对象包装器的一个示例人寿保险应用程序。每个方框表示一个对象,每个对象至少有:

  这些对象与实际的业务逻辑没有关系。对象包装是为了使业务逻辑更容易管理业务数据。包装数据所需的代码比业务逻辑要多得多。更多的代码将导致更多的bug、更大的固定性、更多的维护和更高的成本。

  如果对象中的数据变量可以格式化为XML结构,并且对象的主要作用是将这些数据暴露给业务逻辑并让业务逻辑操纵它们,那么可以用DOM代替对象。

  图2展示了一个使用XML和DOM包装器的示例保险应用程序。图1中的所有数据包装器对象都用一个DOM对象代替。业务数据是用XML建模的,DOM提供了必要的API来:

  串行化和反串行化XML对象层次(换句话说,就是内建持久性)。

  通过使用XML,可以避免使用大多数用于管理业务数据的包装器对象。应用程序将变得更加简洁,并且更多地将重点放在业务逻辑上,而不是数据管理上。

  将XML引入架构中可以为表示业务数据带来一种标准化的方式。XML可以提供数据的结构;XML模式施加结构和格式化规则;DOMAPI和XQuery、XPath及XSLT之类的语言使业务逻辑可以有效地操纵、发布和串行化数据。由于业务数据的XML表示在客户机、中间层和数据库中都是一致的,因此操纵这些对象的代码也是类似的。

  我将展示如何在三层环境中构建基于XML的应用程序,这个三层环境由以下几个部分组成:

  我们来考虑一个简单的人寿保险场景,在这个场景中,首先创建一个表示新保单的XML文档,然后查询和操纵这个文档,另外还将这个文档从一层移动到另一层。这个文档基于合作运营研究与发展协会(AssociationforCooperativeOperationsResearch&Development,ACORD)用于人寿保险规范的XML,它定义了健康保险和年金保险需要交换的数据。

  为了申请一个新的保险,客户需要提供一些基本信息。部分信息是在一个PHP应用程序中填写的,还有一部分是在客户机浏览器中填写的。然后,保单存储在一个DB2XML列中。在DB29中,XML类型的列在内部将XML数据存储为一棵解析后的树,与关系数据存储在不同的地方。这种方法是DB29特有的,更早的DB2版本则使用关系存储基础设施来存储XML。

  Web客户机向PHP发出一个XMLHTTP请求,以获得新的空白保单文档。

  PHP应用程序打开一个空白的保单文档,用一个全局惟一标识符(GUID)更新它,然后将该文档返回给Web客户机。

  Web客户机使用Ajax捕捉返回的事件,并检索XMLDOM,然后用Web页面中输入的信息填充该文档。

  图3展示了用于创建新的保单请求的Web页面。当用户单击Submit按钮后,JavaScript函数submitPolicy()被调用(见清单1)。该函数向PHP应用程序createNewPolicy.php发出一个HTTP请求,以获得一个空白的保单。它还设置一个回调函数fillPolicy(),用于捕捉从HTTP请求返回的事件。

  当第一个请求到达中间层的PHP应用服务器时,一个新的XML保单文档被装载到SimpleXML对象中。通过使用SimpleXMLAPI,用PHP应用程序中创建的GUID更新TransRefGUID元素。

  对于本文,我们假设GUID是通过某种机制(例如时间和随机数的组合)创建的。更重要的是理解如何将表示保单的XML文档视作内存中的业务对象层次结构,以及如何使用SimpleXMLAPI(或DOM/XPath)来导航和更新这个对象。

  在Web客户机中,fillPolicy()函数读取返回的值。现在,包含返回XML的内存中表示的DOM对象可用于操纵保单文档。客户在Web页面上输入的信息被直接用于更新DOM。当使用客户信息更新了保单之后,使用XMLHTTP将修改后的DOM对象提交回PHP应用程序(见清单2)。即使是HTML组件值也是用DHTMLDocumentObjectModel(DOM)读取的。

  PHP应用程序将传入的XML文档直接存储在数据库中,而不需要解析它(见清单3)。DB2的pureXML支持将隐式地解析传入的XML,并将其存储在一个类DOM的分层结构中。现在,可以在XQuery语句中使用XPath之类的XML导航技术(就像在DOM中使用的那样)来查询XML。DB29还提供了在该层次结构中的任何节点上索引的能力。

  新的保单存储在DB29中以后,保险代理可以查询该保单,以决定是否接受这个保险。用于获得关于新保单报告的查询通过Web服务公开给客户机应用程序。

  这个例子中的Web服务是用PHP编写的,它为调用实现服务的业务和转换逻辑的DB2存储过程提供了一个简洁的接口。每个DB2存储过程由一个SQL/XML查询组成,它过滤和转换存储在数据库中的XML保单,以创建一个输出XML文档。然后,该PHPWeb服务将XML文档返回给客户机。

  列出所有新客户保险的DB2查询。包含该查询的存储过程是listAllNewCustomers(见清单4)。该查询搜索ACORD表的INFO列中的所有保单文档。在每个XML文档中,DB2进一步向下分析,以便只返回PolicyStatus/@tc属性的代码值被设为12(也就是建议的值)的那些文档。查询输出是一个XML文档,它有一个根节点newpolicylist,该节点包含一系列的用于每个新保单的TXLife子节点(见图4)。

  注意这个查询第一次如何使用DB2XQuery函数db2-fn:xmlcolumn来导航关系模式,以定位到XML列当它到达XML列时,它进一步使用XPath在XML模式中导航到适当的节点(类似于使用PHP、JavaScript或其他语言导航DOM)。

  列出有风险客户的提议保险的DB2查询。该查询只列出有风险的新客户(也就是说,他们对于某个医疗问题的回答是yes)。该查询包含在一个名为listAtRiskNewCustomers的存储过程中(见清单5)。注意:WHERE子句同时检查答案和保单状态。

  评估有风险新客户的风险度的DB2查询。对于以上列表中的每个保单,在保单的健康风险区域只能列出回答为yes的问题。该查询还返回policytype,以显示该保单值多少钱,以便评估风险。包含该查询的存储过程(见清单6)是getRiskQuestions(guid)。注意:您需要一个支持XML类型的DB2驱动程序版本。否则,在每个存储过程中都需要使用XMLSerialize来从XMLQuery中串行化XML值。请参阅developerWorks文章“结合使用DB2原生XML与PHP”以获得更详细的信息。

  用于getnewpolicyinfoWeb服务的PHP代码是一个瘦包装器,它检查所需保单报告的类型,并调用适当的存储过程。然后,将存储过程所返回的值发送回客户机(见清单7)。注意用PHP创建Web服务是多么简单。最后三行将该功能公开为一个Web服务。在任何客户机中,包括在PHP应用程序中,都可以调用Web服务,如清单8所示。

  在过去几年中,各应用程序层的XML支持都已成熟,这导致一种强大的开发环境的出现,这种开发环境可以改变企业应用的设计方式。XML使开发人员可以为业务文档定义规则和结构,还可以在内存中将文档实例化为分层对象,开发人员可以在任何层使用标准API对这种对象进行导航、修改和串行化。Ajax使基于Web的客户机脚本可以调用DOMAPI,还可以对中间层进行远程过程调用。PHP为处理XML和Web服务提供了最简单的方法之一,因此非常适合基于XML的应用程序开发。XML演变的最后一链是数据库层。DB29使数据库层能操纵XML。因此这个演变周期宣告结束。

相关阅读