PHP调用Webservice实例代码


Posted in PHP onJuly 29, 2011

它是一个开源软件,是完全采用PHP语言编写的、通过HTTP收发SOAP消息的一系列PHP类,由NuSphere Corporation(http://dietrich.ganx4.com/nusoap/ )开发。NuSOAP的一个优势是不需要扩展库的支持,这种特性使得NuSoap可以用于所有的PHP环境,不受服务器安全设置的影响。 

方法一:直接调用

<? 
include(‘NuSoap.php'); 
// 创建一个soapclient对象,参数是server的WSDL 
$client = new soapclient(‘http://localhost/Webservices/Service.asmx?WSDL', ‘wsdl'); 
// 参数转为数组形式传递 
$aryPara = array(‘strUsername'=>'username', ‘strPassword'=>MD5(‘password')); 
// 调用远程函数 
$aryResult = $client->call(‘login',$aryPara); 
//echo $client->debug_str; 

$document=$client->document; 
echo <<<SoapDocument 
<?xml version=”1.0″ encoding=”GB2312″?> 
<SOAP-ENV:Envelope SOAP-ENV:encodingStyle=”http://schemas.xmlsoap.org/soap/encoding/” xmlns:SOAP-ENV=”http://schemas.xmlsoap.org/soap/envelope/” xmlns:xsd=”http://www.w3.org/2001/XMLSchema” xmlns:xsi=”http://www.w3.org/2001/XMLSchema-instance” xmlns:SOAP-ENC=”http://schemas.xmlsoap.org/soap/encoding/” xmlns:si=”http://soapinterop.org/xsd“> 
<SOAP-ENV:Body> 
$document 
</SOAP-ENV:Body> 
</SOAP-ENV:Envelope> 
SoapDocument; 
?>

方法二:代理方式调用
<? 
require(‘NuSoap.php'); 
//创建一个soapclient对象,参数是server的WSDL 
$client=new soapclient(‘http://localhost/Webservices/Service.asmx?WSDL', ‘wsdl'); 
//生成proxy类 
$proxy=$client->getProxy(); 
//调用远程函数 
$aryResult=$proxy->login(‘username',MD5(‘password')); 
//echo $client->debug_str; 

$document=$proxy->document; 
echo <<<SoapDocument 
<?xml version=”1.0″ encoding=”GB2312″?> 
<SOAP-ENV:Envelope SOAP-ENV:encodingStyle=”http://schemas.xmlsoap.org/soap/encoding/” xmlns:SOAP-ENV=”http://schemas.xmlsoap.org/soap/envelope/” xmlns:xsd=”http://www.w3.org/2001/XMLSchema” xmlns:xsi=”http://www.w3.org/2001/XMLSchema-instance” xmlns:SOAP-ENC=”http://schemas.xmlsoap.org/soap/encoding/” xmlns:si=”http://soapinterop.org/xsd“> 
<SOAP-ENV:Body> 
$document 
</SOAP-ENV:Body> 
</SOAP-ENV:Envelope> 
SoapDocument; 
?>

许多使用NuSoap 调用.NET WebService或J2EE WebService的朋友可能都遇到过中文乱码问题,下面介绍这一问题的出现的原因和相应的解决方法。

NuSoap调用WebService出现乱码的原因:

通常我们进行WebService开发时都是用的UTF-8编码,这时我们需要设置:

$client->soap_defencoding = ‘utf-8′;

同时,需要让xml以同样的编码方式传递:

$client->xml_encoding = ‘utf-8′;

至此应该是一切正常了才对,但是我们在输出结果的时候,却发现返回的是乱码。

NuSoap调用WebService出现乱码的解决方法:

实际上,开启了调试功能的朋友,相信会发现$client->response返回的是正确的结果,为什么$result = $client->call($action, array(‘parameters' => $param)); 却是乱码呢?

研究过NuSoap代码后我们会发现,当xml_encoding设置为UTF-8时,NuSoap会检测decode_utf8的设置,如果为true,会执行 PHP 里面的utf8_decode函数,而NuSoap默认为true,因此,我们需要设置:

$client->soap_defencoding = ‘utf-8′;
$client->decode_utf8 = false;
$client->xml_encoding = ‘utf-8′;

补充介绍

NuSOAP 是 PHP 环境下的 WEB 服务编程工具,用于创建或调用 WEB 服务。它是一个开源软件,当前版本是 0.7.2 ,支持 SOAP1.1 、 WSDL1.1 ,可以与其他支持 SOAP1.1 和 WSDL1.1 的系统互操作。 NuSOAP 完全由PHP语言编写,由一系列 PHP 类组成,不需要扩展库的支持,这种特性使得 NuSOAP 可以用于所有的 PHP 环境,不受服务器安全设置的影响。

1. NuSOAP 的获取和安装
NuSOAP 项目建立在 SourceForge 上,网络地址是: http://sourceforge.net/projects/nusoap/ ,这里,可以下载到 NuSOAP 的最新的版本。

NuSOAP 的安装比较简单,把下载的 NuSOAP 的文件拷贝到服务器上,可以放在独立的目录里,也可以与程序代码放在相同的目录里,只要你的 PHP 代码能够访问到这些文件就可以了。
本文的测试环境基于 PHP4.3.2 和 NuSOAP 0.7.2 版本, NuSOAP 安装在 WEB 目录“ /nusoap ”里,有两个子目录, lib 和 samples 。其中, lib 目录下存放 NuSOAP 的所有源代码文件, samples 目录下是NuSOAP开发小组提供一些的例子。测试文件存放在 WEB 目录“ /nusoap ”里。

2. NuSOAP 的使用
NuSOAP 由一 PHP 的类组成,其中最常用到的是类soap_server和类soalclient。类soap_server 用于创建 WEB 服务,类soapclient在访问WEB服务时会用到。
2.1 一个简单的例子: Hello World
这个例子将利用 NuSOAP 创建一个简单的 WEB 服务,并利用 NuSOAP 创建一个客户端程序,调用这个服务。这个服务唯一的功能就是向客户端返回一个字符串“ Hello World ”。首先,创建 WEB 服务程序代码文件“ /nusoap/nusoap_server1.php ”:
//把 NuSOAP 的源文件包含到当前的代码文件里

<?php 
require_once(“lib/nusoap.php”); 
//定义服务程序 
function hello() { 
return ‘Hello World!'; 
} 
//初始化服务对象 , 这个对象是类 soap_server 的一个实例 
$soap = new soap_server; //调用服务对象的 register 方法注册需要被客户端访问的程序。 
//只有注册过的程序,才能被远程客户端访问到。 
$soap->register(‘hello'); //最后一步,把客户端通过 post 方式提交的数据,传递给服务对象的 service 方法。 
//service 方法处理输入的数据,调用相应的函数或方法,并且生成正确的反馈,传回给客户端。 
$soap->service($HTTP_RAW_POST_DATA); 
?>

至此, WEB 服务程序代码文件已经建好,接下来,创建一个客户端程序代码文件“ /nusoap/nusoap_client1.php ”,调用 WEB 服务:
//把 NuSOAP 的源文件包含到当前的代码文件里 
<?php 
require_once(“lib/nusoap.php”); 
//初始化客户端对象,这个对象是类 soapclient 的一个实例, 
//把服务程序的 URL 地址传递给soapclient类的构造函数。 
$client = new soapclient(‘http://127.0.0.1/nusoap/nusoap_server1.php'); //利用客户端对象的 call 方法调用 WEB 服务的程序 
$str=$client->call(‘hello'); //客户端对象的 getError() 方法可以用来检查调用过程是否出现错误。 
//如果没有错误, getError() 方法返回 false ;如果有错误, getError()方法返回错误信息。 
if (!$err=$client->getError()) { 
echo ” 程序返回 :”,htmlentities($str,ENT_QUOTES); 
} else { 
echo ” 错误 :”,htmlentities($err,ENT_QUOTES); 
} 
?>

至此,客户端程序也建立好了,打开浏览器,访问客户端程序,看一下结果。这个例子,浏览器会显示字符串:“程序返回 :Hello World! ”
2.2 传递参数和返回错误信息的方法
再通过例子说明传递参数和返回错误信息的方法。这个例子实现两个字符串的连接,参数是两个字符串,返回值是由两个参数连接而成的字符串。首先,创建服务程序代码文件“ /nusoap/nusoap_server2.php ”,完整的代码如下:
<?php 
require_once(“lib/nusoap.php”); 
function concatenate($str1,$str2) { 
if (is_string($str1) && is_string($str2)) 
return $str1 . $str2; 
else 
return new soap_fault(‘ 客户端 ‘,”,'concatenate 函数的参数应该是两个字符串 ‘); 
} 
$soap = new soap_server; 
$soap->register(‘concatenate'); 
$soap->service($HTTP_RAW_POST_DATA); 
?>

与 2.1 节 WEB 服务程序的代码比较,这里的代码结构大体是相同的。注意以下两点:
服务程序的定义不同,带有两个参数。 NuSOAP 注册服务程序的过程还是一样的,都是调用服务对象的 register 方法。
这里使用了 NuSOAP 的一个新类 soap_fault 。当传入的两个参数有一个不是字符串时,程序通过这个类把错误信息返回给客户端。这个类的构造函数有 4 个参数:
fault
code
必填参数 , 建议值为“ Client ”或“ Server ”,指明错误是客户端的错误还是服务端的错误。

faultactor
预留项,现在还没有使用

faultstring
错误的描述信息

faultdetail
可选项, XML 格式的数据 , 说明详细的错误信息

客户端程序代码文件“ /nusoap/nusoap_client2.php ”的完整内容如下 :

<?php 
require_once(“lib/nusoap.php”); 
$client = new soapclient(‘http://127.0.0.1/nusoap/nusoap_server2.php'); 
$parameters=array(‘ 字符串 1′,' 字符串 2′); 
$str=$client->call(‘concatenate',$parameters); 
if (!$err=$client->getError()) { 
echo ” 程序返回 :”,$str; 
} else { 
echo ” 错误 :”,$err; 
} 
?>

NuSOAP 的客户端调用带参数的 WEB 服务时,使用数组传递参数。 $parameters 是一个数组,其中依次是每个参数的值。客户端在调用远程的服务程序时,使用带有两个参数的 call 方法,第一个参数是服务程序的名称,第二个参数是服务程序的参数数组,这里是 $parameters 。通过浏览器访问上面的客户端程序,浏览器上会显示字符串:“ 程序返回 : 字符串 1 字符串 2 ”
接下来,试着给 WEB 服务程序传入错误参数,修改上面的客户端程序,把生成参数数组的语句改成: $parameters=array(“ 字符串 ”,12) ,再通过浏览器访问客户端程序,浏览器上会显示字符串:“错误 : 客户端 : concatenate 函数的参数应该是两个字符串”。 WEB 服务程序判断传入的参数有一个不是字符串,通过 soap_fault 给客户端返回错误信息。
2.3 调试的方法
NuSOAP中常用的调试方法有三种:
2.3.1 soapclient 类的 request 和 response 成员变量
最直接的调试方法就是检查访问 WEB 服务的过程中,客户端发出的 request 信息和服务端返回的 response 信息。 soapclient 类的 request 和 response 成员变量包含这些信息,在程序中显示出这两个变量的内容,可以帮助分析程序运行的情况。看下面的代码:
<?php 
require_once(“lib/nusoap.php”); 
$client = new soapclient(‘http://127.0.0.1/nusoap/nusoap_server2.php'); 
$parameters=array(‘ 字符串 1′,' 字符串 2′); 
$str=$client->call(‘concatenate',$parameters); 
if (!$err=$client->getError()) { 
echo ” 程序返回 :”,$str; 
} else { 
echo ” 错误 :”,$err; 
} 
//下面显示request和response 变量的内容 
echo ‘<p/>'; 
echo ‘Request:'; 
echo ‘<pre>',htmlspecialchars($client->request,ENT_QUOTES),'</pre>'; 
echo ‘Response:'; 
echo ‘<pre>',htmlspecialchars($client->response,ENT_QUOTES ),'</pre>'; 
?>

2.3.2 soapclient 类的 debug_str 成员变量
soapclient 类的 debug_str 成员变量提供了更为详细的调试信息,查看这个变量的内容,可以更好地帮助程序调试。
2.3.3 WEB 服务程序提供的调试方法
WEB 服务程序代码中,在创建 soap_server 类的实例前,定义变量 $debug=1 。调试信息作为备注,放在 SOAP 消息的尾部返回客户端,客户端通过查看 WEB 服务的 response 信息来查看调试信息。
<?php 
require_once(“lib/nusoap.php”); 
function concatenate($str1,$str2) { 
if (is_string($str1) && is_string($str2)) 
return $str1 . $str2; 
else 
return new soap_fault(‘ 客户端 ‘,”,'concatenate 函数的参数应该是两个字符串 ‘); 
} 
$debug=1; //定义调试 
$soap = new soap_server; 
$soap->register(‘concatenate'); 
$soap->service($HTTP_RAW_POST_DATA); 
?>

2.4 对 WSDL 的支持
NuSOAP 内部通过类 “WSDL” 实现对 WSDL 的支持。对于 NuSOAP 的用户来说,不需要关心内部的WSDL类是如何工作的,正确地使用 soap_server 类和 soapclient 类就可以实现对 WSDL 的支持。
2.4.1 创建支持 WSDL 的 WEB 服务
为了实现 WEB 服务程序对 WSDL 的支持,需要使用 soap_server 的 configureWSDL 方法,并且在调用 soap_server 的 register 方法注册 WEB 服务程序时,需要提供更详细的参数。看下面的代码,代码的文件名是 “/nusoap/nusoap_server3.php”。
<?php 
require_once(“lib/nusoap.php”); 
function concatenate($str1,$str2) { 
if (is_string($str1) && is_string($str2)) 
return $str1 . $str2; 
else 
return new soap_fault(‘ 客户端 ‘,”,'concatenate 函数的参数应该是两个字符串 ‘); 
} 
$soap = new soap_server; 
$soap->configureWSDL(‘concatenate'); // 初始化对 WSDL 的支持 
// 注册服务 
$soap->register(‘concatenate', 
array(“str1″=>”xsd:string”,”str2″=>”xsd:string”), // 输入参数的定义 
array(“return”=>”xsd:string”) // 返回参数的定义 
); 
$HTTP_RAW_POST_DATA = isset($HTTP_RAW_POST_DATA) ? $HTTP_RAW_POST_DATA : ”; 
$soap->service($HTTP_RAW_POST_DATA); 
?>

现在打开浏览器,访问刚才建立的文件,http://127.0.0.1/nusoap/nusoap_server3.php,结果如下:
concatenate
View the WSDL for the service. Click on an operation name to view it's details.
concatenate
点击函数名称concatenate,可以看到对函数的描述。点击”WSDL”,或者访问WEB服务文件,并在后面加上查询字符串”?wsdl”(http://127.0.0.1/nusoap/nusoap_server3.php?wsdl),可以得到WEB服务的WSDL内容。
2.4.2 通过 WSDL 调用 WEB 服务
通过 WSDL 调用 WEB 服务,与不通过 WSDL 调用 WEB 服务,程序的结构大体相同。区别在于,通过 WSDL 调用 WEB 服务,初始化 soapclient 类时,传入两个参数到 soapclient 的构造函数,第一个参数是 WSDL 文件的地址,第二个参数指定是否使用 WSDL ,指定为 true 即可。看下面的代码,代码的文件名是 “ /nusoap/nusoap_client3.php ”
<?php 
require_once(“lib/nusoap.php”); 
$client = new soapclient(‘http://127.0.0.1/nusoap/nusoap_server3.php?wsdl',true); 
$parameters=array(‘ 字符串 1′,' 字符串 2′); 
$str=$client->call(‘concatenate',$parameters); 
if (!$err=$client->getError()) { 
echo ” 程序返回 :”,$str; 
} else { 
echo ” 错误 :”,$err; 
} 
?>

2.4.3 代理的使用
NuSOAP 提供代理的方法调用远程 WEB 服务。这种方法,在客户端程序里面创建一个远程服务的代理对象,通过代理直接调用远程的 WEB 服务,而不需要通过 soalclient 类的 call 方法。看下面的代码。
<?php 
require_once(“lib/nusoap.php”); 
$client = new soapclient(‘http://127.0.0.1/nusoap/nusoap_server3.php?wsdl',true); 
$proxy=$client -> getProxy(); // 创建代理对象 (soap_proxy 类 ) 
$str=$proxy->concatenate(” 参数 1″,” 参数 2″); // 直接调用 WEB 服务 
if (!$err=$proxy->getError()) { 
echo ” 程序返回 :”,$str; 
} else { 
echo ” 错误 :”,$err; 
} 
?>
PHP 相关文章推荐
PHP+JS无限级可伸缩菜单详解(简单易懂)
Jan 02 PHP
php数组函数序列之in_array() 查找数组值是否存在
Oct 29 PHP
php对数组排序的简单实例
Dec 25 PHP
PHP面向对象之旅:深入理解static变量与方法
Jan 06 PHP
PHP内置的Math函数效率测试
Dec 01 PHP
php时间计算相关问题小结
May 09 PHP
php实现的http请求封装示例
Nov 08 PHP
thinkPHP5框架设置404、403等http状态页面的方法
Jun 05 PHP
PHP实现SMTP邮件的发送实例
Sep 27 PHP
laravel 中某一字段自增、自减的例子
Oct 11 PHP
php数值计算num类简单操作示例
May 15 PHP
php优化查询foreach代码实例讲解
Mar 24 PHP
php和数据库结合的一个简单的web实例 代码分析 (php初学者)
Jul 28 #PHP
一个典型的PHP分页实例代码分享
Jul 28 #PHP
新手学习PHP的一些基础知识分享
Jul 27 #PHP
php XPath对XML文件查找及修改实现代码
Jul 27 #PHP
简单的php写入数据库类代码分享
Jul 26 #PHP
php模拟socket一次连接,多次发送数据的实现代码
Jul 26 #PHP
PHP里的中文变量说明
Jul 23 #PHP
You might like
MySQL GBK→UTF-8编码转换
2007/05/24 PHP
收藏的PHP常用函数 推荐收藏保存
2010/02/21 PHP
PHP MemCached高级缓存配置图文教程
2010/08/05 PHP
将一维或多维的数组连接成一个字符串的php代码
2010/08/08 PHP
php模拟ping命令(php exec函数的使用方法)
2013/10/25 PHP
Yii配置文件用法详解
2014/12/04 PHP
分析php://output和php://stdout的区别
2018/05/06 PHP
javascript 类方法定义还是有点区别
2009/04/15 Javascript
Javascript下IE与Firefox下的差异兼容写法总结
2010/06/18 Javascript
判断浏览器的javascript版本的代码
2010/09/03 Javascript
推荐11款jQuery开发的复选框和单选框美化插件
2011/08/02 Javascript
javascript之典型高阶函数应用介绍
2013/01/10 Javascript
node.js中的fs.writeFile方法使用说明
2014/12/14 Javascript
jquery对dom节点的操作【推荐】
2016/04/15 Javascript
Vue 固定头 固定列 点击表头可排序的表格组件
2016/11/25 Javascript
微信小程序使用progress组件实现显示进度功能【附源码下载】
2017/12/12 Javascript
浅谈React + Webpack 构建打包优化
2018/01/23 Javascript
深入理解JS的事件绑定、事件流模型
2018/05/13 Javascript
JavaScript根据json生成html表格的示例代码
2018/10/24 Javascript
js实现鼠标拖拽缩放div实例代码
2019/03/25 Javascript
微信小程序如何调用图片接口API并居中显示
2019/06/29 Javascript
WEEX环境搭建与入门详解
2019/10/16 Javascript
[01:09:20]NB vs NAVI Supermajor小组赛A组 BO3 第二场 6.2
2018/06/03 DOTA
Tornado Web服务器多进程启动的2个方法
2014/08/04 Python
Python实现FTP上传文件或文件夹实例(递归)
2017/01/16 Python
python温度转换华氏温度实现代码
2020/12/06 Python
CSS3毛玻璃效果(blur)有白边问题的解决方法
2016/11/15 HTML / CSS
Carter’s OshKosh加拿大:购买婴幼儿服装和童装
2018/11/27 全球购物
澳大利亚手袋、珠宝和在线时尚精品店:The Way
2019/12/21 全球购物
中学教师实习自我鉴定
2013/09/28 职场文书
历史教育专业个人求职信
2013/12/13 职场文书
团日活动总结
2014/04/28 职场文书
俞敏洪一分钟演讲稿
2014/08/26 职场文书
2014教师党员个人自我评议
2014/09/20 职场文书
光棍节联谊晚会活动策划书
2014/10/10 职场文书
使用Djongo模块在Django中使用MongoDB数据库
2021/06/20 Python