热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

非.NET语言调用.NETXMLWebServices返回的数据集合的两个方法

摘要本文介绍了如何在非.NET语言环境中(如Java、Delphi等)调用.NETXMLWebServices,并特别针对很多开发者所关心的如何在Java
摘要
本文介绍了如何在非.NET语言环境中(如Java、Delphi等)调用.NET XML Web Services,并特别针对很多开发者所关心的"如何在Java/Delphi中使用.NET的Web Service返回的DataSet"的问题,提出了两个解决方案:使用自定义类型数组和使用DataSet.WriteXML()。

目录
  • 1. 概述
  • 2. 创建.NET Web Services,返回数据集合
  • 3. 在Java中调用.NET Web Services,处理返回的数据集合
  • 4. 小结

1.概述
很多正在开发或者打算开发XML Web Services的程序员都问过这样的一个问题:"我的Web Service返回的结果是一个DataSet类型的对象,但如果我的客户端不是用.NET写的(因而没有内建的DataSet类型),那该如何调用这个Web Service并访问DataSet中的数据呢?"。
对于这个问题,首先应该说的是:1)在多种语言共存的编程环境下,是不适合使用类似DataSet这种只属于特定语言的数据类型的。不管是在XML Web Services还是CORBA的环境中,都应该尽量使用简单数据类型以及简单数据类型的数组。2)应当很谨慎的决定是否需要通过
Web Service来返回大量数据。由于网络传输的开销既包括HTTP连接建立的时间,也包括传送数据的时间,因此需要在减少访问服务器次数和减少网络传输量之间寻找一个合适的平衡。如非必须,则不适合通过Web Service传送含有几十条或者几百条数据的数据表。
然后,就问题本身而言,.NET Web Services返回的DataSet类型是可以直接被其他非.NET的客户端解析的,因为即便是DataSet类型的返回值,也会被表达成XML格式再进行传输。下面的例子就是一个返回类型为DataSet的Web Method,及其被调用后返回的XML格式数据:
 
  
[WebMethod]
public  DataSet GetPersonData()
... {
    DataTable table
=new DataTable("Person");        
    table.Columns.Add(
"Name");
    table.Columns.Add(
"Gender");
    table.Rows.Add(
new string[2]...{"Alice","Female"});
    table.Rows.Add(
new string[2]...{"Bob","Male"});
    table.Rows.Add(
new string[2]...{"Chris","Male"});
                
    DataSet dataset
=new DataSet("PersonTable");
    dataset.Tables.Add(table);
         
return dataset;
}

图表1. 返回类型为DataSet的Web Method
 
  
xml version="1.0" encoding="utf-8" ?>
< DataSet  xmlns ="http://tempuri.org/" >
  
< xs:schema  id ="PersonTable"  xmlns =""  xmlns:xs ="http://www.w3.org/2001/XMLSchema"  
    xmlns:msdata
="urn:schemas-microsoft-com:xml-msdata" >
    
< xs:element  name ="PersonTable"  msdata:IsDataSet ="true"  msdata:Locale ="zh-CN" >
      
< xs:complexType >
        
< xs:choice  maxOccurs ="unbounded" >
          
< xs:element  name ="Person" >
            
< xs:complexType >
              
< xs:sequence >
                
< xs:element  name ="Name"  type ="xs:string"  minOccurs ="0"   />
                
< xs:element  name ="Gender"  type ="xs:string"  minOccurs ="0"   />
              
xs:sequence >
            
xs:complexType >
          
xs:element >
        
xs:choice >
      
xs:complexType >
    
xs:element >
  
xs:schema >
  
< diffgr:diffgram  xmlns:msdata ="urn:schemas-microsoft-com:xml-msdata"  
    xmlns:diffgr
="urn:schemas-microsoft-com:xml-diffgram-v1" >
    
< PersonTable  xmlns ="" >
      
< Person  diffgr:id ="Person1"  msdata:rowOrder ="0"  diffgr:hasChanges ="inserted" >
        
< Name > Alice Name >
        
< Gender > Female Gender >
      
Person >
      
< Person  diffgr:id ="Person2"  msdata:rowOrder ="1"  diffgr:hasChanges ="inserted" >
        
< Name > Bob Name >
        
< Gender > Male Gender >
      
Person >
      
< Person  diffgr:id ="Person3"  msdata:rowOrder ="2"  diffgr:hasChanges ="inserted" >
        
< Name > Chris Name >
        
< Gender > Male Gender >
      
Person >
    
PersonTable >
  
diffgr:diffgram >
DataSet >
图表2. 被格式化成XML的DataSet
从上面的例子可以看出,直接使用DataSet作为返回类型,其结果是相当复杂的,其中不但包含了DataSet中的数据,还包括了数据更改的信息,以及DataSet的Schema。虽然有些工具能够生成一个类似DataSet的客户端类型,但无论是直接解析复杂的XML还是使用类似DataSet的类,都不够直接不够清晰。
解决这个问题的方案有两种:
1) 用简单数据类型构造自定义类型,用每一个自定义类型对象封装数据集中的一行,将自定义类型对象的数组(Array)返回客户端;由于是用简单数据类型定义,客户端能够完全不变的还原出自定义类型的定义;
2) 用DataSet.WriteXML()方法将数据集中的数据提取成XML格式,并以字符串的形式返回给客户端,再由客户端解析XML字符串,还原出数据。由于使用WriteXML()的时候能够过滤掉冗余信息,返回的内容和图表2中的内容相比大大简化了。
下面就将用C#分别实现这两种方案,并详细演示如何在Java客户端中调用并获取数据。

2. 创建.NET Web Services,返回数据集合
借助于Visual Studio.NET,只需编写Web Method本身的代码,即可非常快速的创建可以实用的Web Services:
 
  
[WebMethod]
public  Person[] GetPersons()
{
    Person Alice
= new  Person( " Alice " , " Female " );
    Person Bob
= new  Person( " Bob " , " Male " );
    Person Chris
= new  Person( " Chris " , " Female " );
    Person Dennis
= new  Person( " Dennis " , " Male " );

    
return   new  Person[]{Alice,Bob,Chris,Dennis};
}

[WebMethod]
public   string  GetPersonTable()
{
    DataTable table
= new  DataTable( " Person " );        
    table.Columns.Add(
" Name " );
    table.Columns.Add(
" Gender " );
    table.Rows.Add(
new   string [ 2 ]{ " Alice " , " Female " });
    table.Rows.Add(
new   string [ 2 ]{ " Bob " , " Male " });
    table.Rows.Add(
new   string [ 2 ]{ " Chris " , " Female " });
    table.Rows.Add(
new   string [ 2 ]{ " Dennis " , " Male " });
    table.Rows.Add(
new   string [ 2 ]{ " Eric " , " Male " });
    
    DataSet dataset
= new  DataSet( " PersonTable " );
    dataset.Tables.Add(table);
    
    System.Text.StringBuilder strbuilder
= new  System.Text.StringBuilder();
    StringWriter writer
= new  StringWriter(strbuilder);
    dataset.WriteXml(writer,System.Data.XmlWriteMode.IgnoreSchema);
    
    
return  strbuilder.ToString();
}
图表3. 用.NET实现的XML Web Services
 
在上面的代码中,函数GetPersons()和GetPersonTable()分别对应于"1. 概述"中所提到的两种解决方案。其中,Person类型就是用于封装数据集中一行数据的自定义的数据类型:
 
  
[Serializable]
public   class  Person
{
    
public  Person()
    {
    }

    
public  Person( string  name, string  gender)
    {
        
this .Name = name;
        
this .Gender = gender;
    }

    
public   string  Name = "" ;
    
public   string  Gender = "" ;
}
图表4. 自定义类型Person
下面就是在Internet Exploerer里直接调用这两个Web Method所得到的XML格式的结果:
 
  
xml version="1.0" encoding="utf-8" ?>
< ArrayOfPerson  xmlns:xsd ="http://www.w3.org/2001/XMLSchema"  
    xmlns:xsi
="http://www.w3.org/2001/XMLSchema-instance"  
xmlns
="http://tempuri.org/" >
  
< Person >
    
< Name > Alice Name >
    
< Gender > Female Gender >
  
Person >
  
< Person >
    
< Name > Bob Name >
    
< Gender > Male Gender >
  
Person >
  
< Person >
    
< Name > Chris Name >
    
< Gender > Female Gender >
  
Person >
  
< Person >
    
< Name > Dennis Name >
    
< Gender > Male Gender >
  
Person >
ArrayOfPerson >
图表5. GetPersons()的返回结果
 
< PersonTable >   < Person >  
< Name > Alice Name >
 
< Gender > Female Gender >  
Person >   < Person >
 
< Name > Bob Name >
 
< Gender > Male Gender >
 
Person >   < Person >
 
< Name > Chris Name >
 
< Gender > Female Gender >
 
Person >   < Person >
 
< Name > Dennis Name >  
< Gender > Male Gender >  
Person >   < Person >
 
< Name > Eric Name >
 
< Gender > Male Gender >
 
Person >   PersonTable >

图表6. GetPersonTable()的返回的String类型的值
到这里,XML Web Service的创建就已经完成了。.NET中创建的XML Web Services遵循的是统一的行业标准(SOAP、XML等),因此无论使用.NET语言还是使用非.NET语言,都可以调用它。尤其对于GetPersonTable()返回的结果来说,只要用XML Parser作简单的处理,就能把数据提取出来。下面就将演示如何在Java中完成这一处理。

3. 在Java中调用.NET Web Services,处理返回的数据集合
编写客户端调用Web Services时,最快速简便做法的是利用工具根据Web Services的WSDL描述生成客户端代理类(Proxy),而不是直接对SOAP Toolkit进行编程。在.NET Framework中,这个工作可以由wsdl.exe完成。在Java中,可以在JBuilder或者VisualAge Java等Java IDE中安装一个叫做WASP Developer的免费插件,这个插件也能完成由WSDL描述生成Proxy类的工作。另外,该公司还提供了WASP Server for Java和WASP UDDI,降低了用户用Java开发Web Services以及建立UDDI服务的难度。
在本文的例子中,用了JBuilder 7+WASP Developer来调用前文中建立的.NET XML Web Services。在调用之前,首先是用WASP Developer生成Proxy类。从图表7、8中可以看到,生成Proxy的工具已经在Java中也创建了一个Person类:

图表7. 用WASP Developer生成的Proxy类
 
  
package  javaclient.service1Soap12Clientstruct;

public   class  Person {
    
public  java.lang.String Name;
    
public  java.lang.String Gender;
}

/*
 * Generated by WSDLCompiler, (c) 2002, Systinet Corp.
 *                            
http://www.systinet.com
 
*/
图表8. WASP Developer创建的Person类
在Proxy类的基础上,就可以访问.NET编写的Web Services并提取数据了。对于GetPersons()来说,由于已经生成了Person类,所以调用代码非常简单直接:
 
  
import  org.idoox.webservice.client.WebServiceLookup;
import  org.idoox.wasp.Context;
import  javax.swing. * ;
import  javax.swing.table. * ;

try {
  System.setProperty( 
" wasp.location " , " C:/Home/Util/wasp_4.0SP2 " );

  javaclient.service1Soap12Client.Service1Soap service;
  String serviceURI 
=   " http://localhost/dotNetHost/Service1.asmx " ;
  String wsdlURI 
=   " http://localhost/dotNetHost/Service1.asmx?wsdl " ;

  WebServiceLookup lookup 
=  (WebServiceLookup) Context.getInstance(Context.WEBSERVICE_LOOKUP);
  service 
=  (javaclient.service1Soap12Client.Service1Soap) lookup.lookup(
            wsdlURI, 
            javaclient.service1Soap12Client.Service1Soap.
class
            serviceURI);

  Person[] persons
=
   service.GetPersons(
new  javaclient.service1Soap12Clientstruct.GetPersons()).GetPersonsResult;
  DefaultTableModel model
= new  DefaultTableModel( new  String[]{ " name " , " gender " }, 0 );
  
for ( int  i = 0 ;i < persons.length;i ++ ){
    model.addRow(
new  String[]{persons[i].Name.toString(),persons[i].Gender.toString()});
  }
  
this .jTable1.setModel(model);
}
catch (Exception ex)
{
  javax.swing.JOptionPane.showMessageDialog(
this ,ex.getMessage());
}
图表9. 在Java中调用GetPersons()的代码
调用GetPersonTable()并解析XML字符串要相对复杂一些,需要用到XML Parser从图表6中的XML字符串中提取数据,并逐一显示在JTable中。下面是从访问Web Method到解析XML,最后提取数据并显示的完整代码,及其运行结果截图:
import  javax.swing. * ;
import  javax.swing.table. * ;

import  org.idoox.webservice.client.WebServiceLookup;
import  org.idoox.wasp.Context;

import  org.w3c.dom. * ;
import  org.apache.xml.serialize. * ;
import  org.apache.xerces.dom. * ;

try {
  System.setProperty( 
" wasp.location " , " C:/Home/Util/wasp_4.0SP2 " );

  javaclient.service1Soap12Client.Service1Soap service;
  String serviceURI 
=   " http://localhost/dotNetHost/Service1.asmx " ;
  String wsdlURI 
=   " http://localhost/dotNetHost/Service1.asmx?wsdl " ;

  WebServiceLookup lookup 
=  (WebServiceLookup) Context.getInstance(Context.WEBSERVICE_LOOKUP);
  service 
=  (javaclient.service1Soap12Client.Service1Soap) lookup.lookup(
            wsdlURI, 
            javaclient.service1Soap12Client.Service1Soap.
class
            serviceURI);

  String result
=
   service.GetPersonTable(
new  javaclient.service1Soap12Clientstruct.GetPersonTable()).GetPersonTableResult;
  DefaultTableModel model
= new  DefaultTableModel( new  String[]{ " name " , " gender " }, 0 );

  java.io.StringReader reader
= new  java.io.StringReader(result);
  org.apache.xerces.parsers.DOMParser parser
= new  org.apache.xerces.parsers.DOMParser();
  parser.parse(
new  org.xml.sax.InputSource(reader));
  org.w3c.dom.Document document
= parser.getDocument();
  org.w3c.dom.NodeList children
= document.getDocumentElement().getChildNodes();

  
for ( int  i = 0 ;i < children.getLength();i ++ ){
    org.w3c.dom.Node node
= children.item(i);
    
if (node.getNodeName().equals( " Person " )){
      org.w3c.dom.NodeList personAttrNodes
= node.getChildNodes();
      String name
= " N/A " ;
      String gender
= " N/A " ;

      
for ( int  j = 0 ;j < personAttrNodes.getLength();j ++ ){
        org.w3c.dom.Node attrNode
= personAttrNodes.item(j);
        String attrNodeName
= attrNode.getNodeName();
        
if (attrNodeName.equals( " Name " )){
          name
= attrNode.getFirstChild().getNodeValue();
        }
else   if (attrNodeName.equals( " Gender " )){
          gender
= attrNode.getFirstChild().getNodeValue();
        }
else {
          
continue ;
        }
      }
      model.addRow(
new  String[]{name,gender});
    }
  }
  
this .jTable1.setModel(model);
}
catch (Exception ex)
{
  javax.swing.JOptionPane.showMessageDialog(
this ,ex.getMessage());
}
图表10. 在Java中调用GetPersonTable()的代码

图表11.调用GetPersonTable()的运行结果

4. 小结
从前面的叙述和代码中可以看出,对于"如何在Java/Delphi中使用.NET的Web Service返回的DataSet"的问题,虽然在非.NET语言环境中直接接受DataSet类型的返回值比较困难,但可以有其他的解决方案。
对于第一种解决方案,也就是采用自定义数据类型,它的优点是客户端代码非常简单,容易编写,而且容易理解。它的缺点是服务器端需要定义新的类,而且当通过ADO.NET从数据库中提取数据以后,还要再手工编写代码,将DataSet中的数据遍历一遍,转存成自定义类型的对象。
对于第二种解决方案,就是用DataSet.WriteXML()的方案,它的优点和缺点正好和第一种方案是互补的。它在客户端需要编写较多的代码,尤其是需要用XML Parser编程,提高了编程的难度。另一方面,它在服务器端比较简单。如果开发者在原先的纯.NET环境(即客户端也是.NET)中已经开发了返回类型为DataSet的Web Services,那么当开发者希望在客户端引入非.NET语言时,服务器端只需要简单的再增加一个Web Method,通过简单的WriteXML()调用对原有的方法进行简单的包装,就能够适用于非.NET客户端了。
从更广阔的角度来看,上面两种解决方案也可以用在"非.NET服务器+.NET客户端"的Web Services环境中。在各种非原生(Native)的情况下,尽量使用简单数据类型,或者使用XML来表达数据,总是上佳的选择。

作者:郑子颖
  http://www.microsoft.com/china/community/program/originalarticles/techdoc/callwebs.mspx

推荐阅读
  • 本文详细介绍了Python中的流程控制与条件判断技术,包括数据导入、数据变换、统计描述、假设检验、可视化以及自定义函数的创建等方面的内容。 ... [详细]
  • Golang与微服务架构:构建高效微服务
    本文探讨了Golang在微服务架构中的应用,包括Golang的基本概念、微服务开发的优势、常用开发工具以及具体实践案例。 ... [详细]
  • 在尝试使用 Android 发送 SOAP 请求时遇到错误,服务器返回 '无法处理请求' 的信息,并指出某个值不能为 null。本文探讨了可能的原因及解决方案。 ... [详细]
  • 本文作者分享了在某大型IToIP解决方案提供商参与多个项目开发的经验与感悟,特别是在软件工程方法论上的思考,提出了对现有开发模式的见解及改进建议。 ... [详细]
  • 本文探讨了如何利用 Application 对象在 PHP 应用程序中共享数据,特别是在多用户环境中保持数据的一致性和安全性。文章还介绍了 Application 对象的基本结构、方法和事件,并提供了实际应用示例。 ... [详细]
  • 本文探讨了一种方法,通过开发C#应用程序来拦截并处理从遗留系统发出的Http请求,该系统原本依赖于已停止服务的Web服务。解决方案涉及使用代理技术或HTTP监听器来捕获和重定向这些请求。 ... [详细]
  • 本文探讨了在JavaScript中如何有效地从服务器控件DropDownList中获取绑定的ID值,而非仅仅是显示的文本值。这对于需要根据用户选择动态处理数据的应用场景非常有用。 ... [详细]
  • 2023年PHP处理请求超时的全面指南
    本文详细介绍了在PHP中处理请求超时的各种方法,包括设置脚本执行时间、处理file_get_contents函数超时以及优化AJAX请求等,适合开发者参考学习。 ... [详细]
  • 浏览器、中间件与服务器的交互机制
    本文详细探讨了浏览器、中间件和服务器之间的交互过程,特别是HTTP请求的完整流程,包括DNS解析、TCP连接建立及数据传输等关键步骤。 ... [详细]
  • 深入理解SAP Fiori及其核心概念
    本文详细介绍了SAP Fiori的基本概念、发展历程、核心特性、应用类型、运行环境以及开发工具等,旨在帮助读者全面了解SAP Fiori的技术框架和应用场景。 ... [详细]
  • 匠心|传统_2021年度总结 | 葡萄城软件开发技术回顾(上)
    匠心|传统_2021年度总结 | 葡萄城软件开发技术回顾(上) ... [详细]
  • 本文详细介绍了Python的multiprocessing模块,该模块不仅支持本地并发操作,还支持远程操作。通过使用multiprocessing模块,开发者可以利用多核处理器的优势,提高程序的执行效率。 ... [详细]
  • 本文深入探讨了分布式文件系统的核心概念及其在现代数据存储解决方案中的应用,特别是针对大规模数据处理的需求。文章不仅介绍了多种流行的分布式文件系统和NoSQL数据库,还提供了选择合适系统的指导原则。 ... [详细]
  • PHP 5.5.31 和 PHP 5.6.17 安全更新发布
    PHP 5.5.31 和 PHP 5.6.17 已正式发布,主要包含多个安全修复。强烈建议所有用户尽快升级至最新版本以确保系统安全。 ... [详细]
  • REST与RPC:选择哪种API架构风格?
    在探讨REST与RPC这两种API架构风格的选择时,本文首先介绍了RPC(远程过程调用)的概念。RPC允许客户端通过网络调用远程服务器上的函数或方法,从而实现分布式系统的功能调用。相比之下,REST(Representational State Transfer)则基于资源的交互模型,通过HTTP协议进行数据传输和操作。本文将详细分析两种架构风格的特点、适用场景及其优缺点,帮助开发者根据具体需求做出合适的选择。 ... [详细]
author-avatar
伍贤厚_197
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有