php weather 客户机
这一节将建立我们自己的 php weather 客户机。这里提供了一些代码片段,建议下载完整的客户机和 wsdl 文件。
用于表示 weather service 的 ext/soap 类是 soapclient。正如我们介绍 weather forecast 应用程序时所讨论的,我们知道应用服务器在 http://host:port/itsowebserver2routerweb/wsdl/itso/session/weatherforecastejb.wsdl 中提供了 wsdl。我们使用的是默认端口,并且在作为服务器的计算机上工作,这样就可以通过查找 wsdl 文件创建第一个 soapclient:
<?php $soapclient = new soapclient(“http://localhost:9080/” . “itsowebservice2routerweb/wsdl/itso/session/weatherforecastejb.wsdl”); ?> |
注意,因为 ext/soap 是内置的,所以,在引用 soapclient 之前,不需要任何 include 或 require 语句。
现在已经实例化了客户机,还要联系 weather 服务,并调用它的 getforecast 操作。在 wsdl 模式下使用 soapclient 时,ext/soap 有一种很好的特性,即可以直接引用远程操作,就像它是 soapclient 自身的函数一样。但是在建立输入参数时需要一点技巧。ext/soap 可以提供从 wsdl 中发现的操作和参数的数组:
$functions = $soapclient->__getfunctions(); print_r($functions); $types = $soapclient->__gettypes(); print_r($types); |
只需要显示与 getforecast 相关的结果,并重新格式化这些结果,以方便阅读,于是我们看到以下代码:
getforecastresponse getforecast(getforecast $parameters)
struct getforecast { struct getforecastresponse { struct weather { |
ext/soap 实际上并没有为我们定义 getforecast 类,我们必须创建该操作所需要的输入参数数组:
$getforecastparam = array(startdate =>time(), days => 3); |
然后像 soapclient 的方法那样调用该操作:
$forecastresponse = $soapclient->getforecast($getforecastparam); |
最后我们得到了返回的 getforecastresponse 对象,它本身是一个 weather 对象数组,然后在表格中显示结果:
echo “<table border=1 cellpadding=5>”; echo “<tr><th>date</th><th>condition</th><th>temperature</th><th>wind</th></tr>”; $weatherarray = $forecastresponse->getforecastreturn; foreach ($weatherarray as $weather) { echo “<tr>”, “<td>”,strftime(“%a. %b %d, %y”, strtotime($weather->date)),”</td>”, “<td>$weather->condition</td>”, “<td>$weather->temperaturecelsius</td>”, “<td>$weather->winddirection $weather->windspeed</td>”, “</tr>”; } echo “</table>”; |
php 客户机与 java 客户机的输出相同,于是我们知道圣诞节期间 san jose 不会下雪……
图 3. php weatherclient
|
观察 soap 流
我们成功地与 weather 服务取得了联系,并显示了结果。但是如果出现错误,得不到预期的结果,该怎么办?ext/soap 可以显示客户机与服务器之间交换的 soap 消息,能够帮助我们确定问题所在。
只有使用 trace 选项创建 soapclient 时,才要使用跟踪功能。我们在 options 数组参数中设置 trace 选项,将该参数传递给 soapclient 构造函数。我们将构造函数的使用改为:
$soapclient = new soapclient(“http://localhost:9080/” . “itsowebservice2routerweb/wsdl/itso/session/weatherforecastejb.wsdl”, array(trace => 1)); |
并在调用 goforecast 之后调用 trace 方法:
echo “request :<br>”, htmlspecialchars($soapclient->__getlastrequest()), “<br>”; echo “response :<br>”, htmlspecialchars($soapclient->__getlastresponse()), “<br>”; |
一定要使用 htmlspecialchars 内置函数对 trace 输出进行编码,因为它将 soap xml 分界符转换成特殊字符,如 <,这样可以避免浏览器将其解释成标记。
下面是某个请求的 trace 输出:
<?xml version=”1.0″ encoding=”utf-8″?> <soap-env:envelope xmlns:soap-env=”http://schemas.xmlsoap.org/soap/envelope/” xmlns:ns1=”http://session.itso”> <soap-env:body> <ns1:getforecast> <ns1:startdate>2004-11-30t13:41:59</ns1:startdate> <ns1:days>0</ns1:days> </ns1:getforecast> </soap-env:body> </soap-env:envelope> |
对应的应答是:
<?xml version=”1.0″ encoding=”utf-8″?> <soapenv:envelope xmlns:soapenv=”http://schemas.xmlsoap.org/soap/envelope/” xmlns:soapenc=”http://schemas.xmlsoap.org/soap/encoding/” xmlns:xsd=”http://www.w3.org/2001/xmlschema” xmlns:xsi=”http://www.w3.org/2001/xmlschema-instance”> <soapenv:body> <getforecastresponse xmlns=”http://session.itso”> <getforecastreturn xmlns:ns-239399687=”http://mapping.itso”> <ns-239399687:condition>sunny</ns-239399687:condition> <ns-239399687:date>2004-11-30t00:00:00.000z</ns-239399687:date> <ns-239399687:winddirection>w</ns-239399687:winddirection> <ns-239399687:windspeed>18</ns-239399687:windspeed> <ns-239399687:temperaturecelsius>6</ns-239399687:temperaturecelsius> <ns-239399687:dbflag>1</ns-239399687:dbflag> </getforecastreturn> </getforecastresponse> </soapenv:body> </soapenv:envelope> |
如果在开启跟踪功能的情况下运行客户机来收集这些输出,那么需要将 days 参数设置为 0,只有这样做,soap 应答才会输出较少的行。但是我们遇到了没有预料到的行为。我们本来期望 getforecastresponse 和以前一样是一个 weather 对象数组,但是它应该只有一个元素,而不是 4 个元素。然而,它被转换成了一个简单的 weather 对象,我们必须根据这种行为进行编码,就像您在最终的示例 php 客户机代码中看到的那样。这与 java 客户机的行为有所不同,在客户机行为中,getforecast 总是返回 weather 对象数组,无论服务器响应中有多少个 weather 对象。soapclient::_gettypes() 输出并没有为我们理解这种差异提供足够的细节,因此我们要求助于 wsdl 文档来了解完整的接口规范。