JAVA与.NET是现今世界竞争激烈的两大开发媒体,两者语言有很多相似的地方。而在很多大型的开发项目里面,往往需要使用两种语言进行集成开 发。而很多的开发人员都会偏向于其中一种语言,在使用集成开发的时候对另一种语言感觉到畏惧。在这里在下向各位介绍一下,JAVA与.NET相互调用的例 子。下面的介绍主要包括三方面:一是通过常用Web服务进行相互调用,二是使用TCP/IP套接字进行相互调用,三是使用Remote实现远程对象相互调 用。

在这章里面先为大家介绍一下最简单,最常用的Web服务相互调用方式。首先说一下Web服务的来源,Web服务是一种新的Web应用程序分支,可以 执行从简单的请求到复杂商务处理等任何功能。一旦部署以后,其他Web服务应用程序可以发现并调用它部署的服务。 Web Service是一种应用程序,它可以使用标准的互联网协议,像超文件传输协议(HTTP)、简单对象访问协议(SOAP)、XML等,将功能纲领性地体 现在互联网和企业内部网上,Web服务被视作是Web上的组件编程。Web服务必须提供一套标准的类型系统,用于沟通不同平台、编程语言和组件模型中的不 同类型系统。

XML和XSD

可扩展的标记语言XML 是Web Service平台中表示数据的基本格式。除了易于建立和易于分析外,XML主要的优点在于它既与平台无关,又与厂商无关。XML是由万维网协会 (W3C)创建,W3C制定的XML SchemaXSD 定义了一套标准的数据类型,并给出了一种语言来扩展这套数据类型。 Web Service平台是用XSD来作为数据类型系统的。当你用某种语言如JAVA、C#来构造一个Web Service时,为了符合Web Service标准,所有你使用的数据类型都必须被转换为XSD类型。如想让它使用在不同平台和不同软件的不同组织间传递,还需要通过SOAP协议将它包 装起来。

SOAP

SOAP即简单对象访问协议(Simple Object Access Protocol),它是用于交换XML编码信息的轻量级协议。它有三个主要方面:XML-envelope为描述信息内容和如何处理内容定义了框架,将 程序对象编码成为XML对象的规则,执行远程过程调用(RPC)的约定。SOAP可以运行在任何其他传输协议上。例如,你可以使用 SMTP,即因特网电子邮件协议来传递SOAP消息,这可是很有诱惑力的。在传输层之间的头是不同的,但XML有效负载保持相同。Web Service 希望实现不同的系统之间能够用“软件-软件对话”的方式相互调用,打破了软件应用、网站和各种设备之间的格格不入的状态,实现“基于Web无缝集成”的目 标。

WSDL

Web Service描述语言WSDL 就是用机器能阅读的方式提供的一个正式描述文档而基于XML的语言,用于描述Web Service及其函数、参数和返回值。因为是基于XML的,所以WSDL既是机器可阅读的,又是人可阅读的。

下面分开两个方面讲解一下如果通过Web服务实现JAVA与.NET的相互调用。

 

一、使用.NET作为服务器端,JAVA作为客户端实现相互调用。

在.NET系统里面,以WCF作为新一代的服务开发工具是微软的一个新卖点,我们就以WCF为例子实现服务器端,首先新建一个网站项目,在网站添加 一个WCF服务PersonService。你将看到PersonService.svc、IPersonService、 PersonService.cs三个文件,其中IPersonService是对向暴露一个接口,接口的功能由PersonService来实现,客户 端则通过PersonalService.svc来寻获服务,并对其添加引用的。


   
   
  1. //在PersonService.svc里,只包括一行,其中列明了该服务的实现类 
  2. <%@ ServiceHost Language="C#" Debug="true" Service="Service.PersonService"
  3. CodeBehind="~/App_Code/PersonService.cs" %> 
  4.  
  5. // 注意: 使用“重构”菜单上的“重命名”命令,可以同时更改代码和
  6. // 配置文件中的接口名“IPersonService”。 
  7. namespace Service 
  8.     [ServiceContract] 
  9.     public interface IPersonService 
  10.     { 
  11.         [OperationContract] 
  12.         IList<Person> GetList(); 
  13.     } 
  14.  
  15.     public class PersonService : IPersonService 
  16.     { 
  17.         public IList<Person> GetList() 
  18.         { 
  19.             IList<Person> personList = new List<Person>(); 
  20.              
  21.             Person person1 = new Person(); 
  22.             person1.ID = 0; 
  23.             person1.Age = 27; 
  24.             person1.Name = "Leslie"
  25.             personList.Add(person1); 
  26.              
  27.             Person person2 = new Person(); 
  28.             person2.ID = 1; 
  29.             person2.Age = 23; 
  30.             person2.Name = "Rose"
  31.             personList.Add(person2); 
  32.  
  33.             Person person3 = new Person(); 
  34.             person3.ID = 2; 
  35.             person3.Age = 29; 
  36.             person3.Name = "Jack"
  37.             personList.Add(person3); 
  38.              
  39.             return personList; 
  40.         } 
  41.     } 

为了使用Person能够实现远程传送,我们必须对Person进行序列化,在WCF中包括服务契约、数据契约、消息契约三部分,而数据契约正是用于对数据进行序列化处理的,如果想对WCF有进一步的了解,可以链接使用WCF实现SOA面向服务编程


   
   
  1. namespace Service 
  2.     //建立Person数据契约 
  3.     [DataContract] 
  4.     public class Person 
  5.     { 
  6.         [DataMember] 
  7.         public int ID 
  8.         { 
  9.             get
  10.             set
  11.         } 
  12.         [DataMember] 
  13.         public string Name 
  14.         { 
  15.             get
  16.             set
  17.         } 
  18.         [DataMember] 
  19.         public int Age 
  20.         { 
  21.             get
  22.             set
  23.         } 
  24.     } 

数据契约里面有多种的序列化方式,包括 DataContractSerializer,NetDataContractSerializer,XmlServializer,DataContractJsonSerializer。 在这里面只用使用最普遍的DataContractSerializer,而DataContractJsonSerializer是现今比较热门的方 式,特别是在开发网络项目时候,多使用Json进行数据通讯。

最后配置好web.config,就可以成功将WCF服务发布


   
   
  1. <?xml version="1.0"?> 
  2. <configuration> 
  3.     <system.web> 
  4.         <compilation debug="true" targetFramework="4.0"> 
  5.             <assemblies> 
  6.                 <add assembly="System.Data.Entity, Version=4.0.0.0, Culture=neutral
  7.              PublicKeyToken=B77A5C561934E089"/> 
  8.       </assemblies></compilation> 
  9.     </system.web> 
  10.     <system.serviceModel> 
  11.         <behaviors> 
  12.             <serviceBehaviors> 
  13.                 <behavior name="serviceBehavior"> 
  14.                     <!--注意将httpGetEnabled设置为true,使客户端能够成功捕获服务--> 
  15.                     <serviceMetadata httpGetEnabled="true"/> 
  16.                     <serviceDebug includeExceptionDetailInFaults="false"/> 
  17.                 </behavior> 
  18.             </serviceBehaviors> 
  19.         </behaviors> 
  20.     <services> 
  21.        <!--name属性必须与服务实现类的类名相对应--> 
  22.       <service name="Service.PersonService" behaviorConfiguration="serviceBehavior"> 
  23.         <!--contract必须与契约名相对应--> 
  24.         <endpoint address="" binding="basicHttpBinding" contract="Service.IPersonService"/> 
  25.          <!--注意打开元数据,使客户能下载--> 
  26.         <endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange"/> 
  27.       </service> 
  28.     </services> 
  29.     </system.serviceModel> 
  30. </configuration> 

下面使用MyEclipse8.6进行客户端开发,首先添加对服务的引用,按Ctrl+N新建一个项目,选择Web Service->Web Service Client,单击下一步,这时候选择在Framework上选择JAX-WS,单击下一步

在WSDL URL上输入服务的路径,并为服务添加一个Java pagckage包myServices,点击完成,这样WCF服务便可成功加入到客户端。

此时为此项目添加测试类,运行进行测试


   
   
  1. package myAssembly; 
  2.  
  3. import java.util.List; 
  4.  
  5. public class Test { 
  6.    public static void main(String[] args){ 
  7.        myServices.PersonService service=new myServices.PersonService(); 
  8.        //获取服务对象 
  9.        myServices.IPersonService personService= 
  10.            service.getBasicHttpBindingIPersonService(); 
  11.        //通过basicHttpBinding协议绑定远程对象 
  12.         
  13.        List<myServices.Person> personList=personService.getList().getPerson(); 
  14.     
  15.        for(int n=0;n<personList.size();n++){ 
  16.            System.out.println("ID:"+personList.get(n).getID()+ 
  17.                    " Name:"+personList.get(n).getName().toString()+ 
  18.                    " Age:"+personList.get(n).getAge()); 
  19.        } 
  20.    } 

 

二、使用JAVA作为服务器端,.NET作为客户端实现相互调用。

JAVA开发Web Service的工具有很多,最常用的有Axis、XFire、NetBean等,在JAVA-SE 6.0以上支持JAX-WS2.0 ,JAX-WS 2.0是JAX-RPC 1.0的更新产品。在 JAX-WS中,一个远程调用可以转换为一个基于XML的协议例如SOAP。在使用JAX-WS过程中,开发者不需要编写任何生成和处理SOAP消息的代 码。JAX-WS的运行时实现会将这些API的调用转换成为对于SOAP消息。 在服务器端,用户只需要通过Java语言定义远程调用所需要实现的接口SEI (service endpoint interface),并提供相关的实现,通过调用JAX-WS的服务发布接口就可以将其发布为WebService接口。在下面我们就以XFire建立 一个Web Service。

首先建立一个在一个项目上单击右键,选择MyEclipse->Add XFire Web Service Capabilities,引用了XFire工具包以后。在项目会自动建立一个WebServices文件夹,文件夹里面的service.xml就是对 发布Web Service进行配置的。

现在先建立好一个服务层


   
   
  1. //建立一个Model包,里面包含一个值对象Person 
  2. package Model; 
  3.  
  4. import java.io.Serializable; 
  5. public class Person implements Serializable { 
  6.     private int id; 
  7.     private String name; 
  8.     private int age; 
  9.      
  10.     public int getId(){ 
  11.         return id; 
  12.     } 
  13.      
  14.     public void setId(int id){ 
  15.         this.id=id; 
  16.     } 
  17.      
  18.     public String getName(){ 
  19.         return name; 
  20.     } 
  21.      
  22.     public void setName(String name){ 
  23.         this.name=name; 
  24.     } 
  25.      
  26.     public int getAge(){ 
  27.         return age; 
  28.     } 
  29.      
  30.     public void setAge(int age){ 
  31.         this.age=age; 
  32.     } 
  33.  
  34. //建立一个Service包,里面包含服务接口 
  35. package Service; 
  36.  
  37. import java.util.List; 
  38. import Model.*; 
  39.  
  40. public interface PersonService { 
  41.    List<Person> GetList(); 
  42.  
  43. //建立一个ServiceImpl包,实现服务 
  44. package ServiceImpl; 
  45.  
  46. import Model.*; 
  47. import Service.*; 
  48. import java.util.*; 
  49.  
  50. public class PersonServiceImpl implements PersonService{ 
  51.    public List<Person> GetList(){ 
  52.        List<Person> personList=new LinkedList<Person>(); 
  53.         
  54.        Person person1=new Person(); 
  55.        person1.setId(0); 
  56.        person1.setAge(23); 
  57.        person1.setName("Leslie"); 
  58.        personList.add(person1); 
  59.         
  60.        Person person2=new Person(); 
  61.        person2.setId(1); 
  62.        person2.setAge(30); 
  63.        person2.setName("Mike"); 
  64.        personList.add(person2); 
  65.         
  66.        return personList; 
  67.    } 

在Service.xml上面对服务进行配置


   
   
  1. <?xml version="1.0" encoding="UTF-8"?> 
  2. <beans xmlns="http://xfire.codehaus.org/config/1.0"> 
  3. <service> 
  4.     <name>PersonService</name> 
  5.     <namespace>http://leslie-pc:8080/PersonService</namespace> 
  6.     <serviceClass> 
  7.         Service.PersonService 
  8.     </serviceClass> 
  9.     <implementationClass> 
  10.         ServiceImpl.PersonServiceImpl 
  11.     </implementationClass> 
  12. </service> 
  13. </beans> 

其配置功能如下:

  • service

    service 标签和它所包含的 xml 内容为发布成 Web 服务的 POJO 提供完整的描述。

  • name

    Web 服务被发布时所采用的唯一名称。

  • namespace

    Web 服务发布时所使用的命名空间。

  • serviceClass

    Web 服务接口类的全名,包括包名和类名。

  • implemetationClass

    Web 服务实现类的全名,包括包名和类名。

现在可以运行程序,对服务进行测试,在测试时输入服务地址http://leslie-pc:8080/WebSite1/services/PersonService?wsdl,系统将显示wsdl代码


   
   
  1. <?xml version="1.0" encoding="UTF-8" ?> 
  2. <wsdl:definitions targetNamespace="http://leslie-pc:8080/PersonService" xmlns:ns1="http://Model" xmlns:soapenc12="http://www.w3.org/2003/05/soap-encoding" xmlns:tns="http://leslie-pc:8080/PersonService" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:soap11="http://schemas.xmlsoap.org/soap/envelope/" xmlns:wsdlsoap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:soapenc11="http://schemas.xmlsoap.org/soap/encoding/" xmlns:soap12="http://www.w3.org/2003/05/soap-envelope"> 
  3. <wsdl:types> 
  4. <xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema" attributeFormDefault="qualified" elementFormDefault="qualified" targetNamespace="http://leslie-pc:8080/PersonService"> 
  5. <xsd:element name="GetList"> 
  6. <xsd:complexType /> 
  7. </xsd:element> 
  8. <xsd:element name="GetListResponse"> 
  9. <xsd:complexType> 
  10. <xsd:sequence> 
  11. <xsd:element maxOccurs="1" minOccurs="1" name="out" nillable="true" type="ns1:ArrayOfPerson" /> 
  12. </xsd:sequence> 
  13. </xsd:complexType> 
  14. </xsd:element> 
  15. </xsd:schema> 
  16. <xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema" attributeFormDefault="qualified" elementFormDefault="qualified" targetNamespace="http://Model"> 
  17. <xsd:complexType name="ArrayOfPerson"> 
  18. <xsd:sequence> 
  19. <xsd:element maxOccurs="unbounded" minOccurs="0" name="Person" nillable="true" type="ns1:Person" /> 
  20. </xsd:sequence> 
  21. </xsd:complexType> 
  22. <xsd:complexType name="Person"> 
  23. <xsd:sequence> 
  24. <xsd:element minOccurs="0" name="age" type="xsd:int" /> 
  25. <xsd:element minOccurs="0" name="id" type="xsd:int" /> 
  26. <xsd:element minOccurs="0" name="name" nillable="true" type="xsd:string" /> 
  27. </xsd:sequence> 
  28. </xsd:complexType> 
  29. </xsd:schema> 
  30. </wsdl:types> 
  31. <wsdl:message name="GetListRequest"> 
  32. <wsdl:part name="parameters" element="tns:GetList" /> 
  33. </wsdl:message> 
  34. <wsdl:message name="GetListResponse"> 
  35. <wsdl:part name="parameters" element="tns:GetListResponse" /> 
  36. </wsdl:message> 
  37. <wsdl:portType name="PersonServicePortType"> 
  38. <wsdl:operation name="GetList"> 
  39. <wsdl:input name="GetListRequest" message="tns:GetListRequest" /> 
  40. <wsdl:output name="GetListResponse" message="tns:GetListResponse" /> 
  41. </wsdl:operation> 
  42. </wsdl:portType> 
  43. <wsdl:binding name="PersonServiceHttpBinding" type="tns:PersonServicePortType"> 
  44. <wsdlsoap:binding style="document" transport="http://schemas.xmlsoap.org/soap/http" /> 
  45. <wsdl:operation name="GetList"> 
  46. <wsdlsoap:operation soapAction="" /> 
  47. <wsdl:input name="GetListRequest"> 
  48. <wsdlsoap:body use="literal" /> 
  49. </wsdl:input> 
  50. <wsdl:output name="GetListResponse"> 
  51. <wsdlsoap:body use="literal" /> 
  52. </wsdl:output> 
  53. </wsdl:operation> 
  54. </wsdl:binding> 
  55. <wsdl:service name="PersonService"> 
  56. <wsdl:port name="PersonServiceHttpPort" binding="tns:PersonServiceHttpBinding"> 
  57. <wsdlsoap:address location="http://leslie-pc:8080/WebSite1/services/PersonService" /> 
  58. </wsdl:port> 
  59. </wsdl:service> 
  60. </wsdl:definitions> 

服务器端测试已经成功,现在使用.NET对服务进行调用,在项目上单击右键->选择添加Web服务->在URL地址上输入服务的地址 http://leslie-pc:8080/WebSite1/services/PersonService?wsdl  ,在一个页面上输入代码进行测试。


   
   
  1. protected void Page_Load(object sender, EventArgs e) 
  2.     Service.PersonService personService = new Service.PersonService(); 
  3.     IList<Service.Person> personList = personService.GetList(); 
  4.     foreach(Service.Person person in personList) 
  5.     { 
  6.           Response.Write("id:" + person.id.ToString() + " name:" +  
  7.            person.name + " age:" + person.age.ToString()+"<br/>"); 
  8.     } 

测试成功的话,恭喜你,你已经了解到JAVA与.NET是如何通过Web服务进行相互调用的了。但因为Web服务从本质是就是不受开发语言的局限 的,所以只要阁下对JAVA跟.NET有一定了解,要通过Web服务实现相互调用相信不是一个难题。但往往在一些ERP,OA的开发过程,会在很多时候使 用TCP/IP套接字实现软件的功能,TCP/IP这“老家伙”为何使用了这么长时间还会经常见到它的身影,这是因为使用TCP/IP有着更高效率,而且 易于通过防火墙的阻隔,而HTTP协议也是建立一TCP/IP之上的。在下一章将为大家介绍JAVA与.NET是如何通过TCP/IP套接字进行相互调用 的。

 

原代码: (由于上传空间有限,未能将JAVA项目的.metadata一并上传,请运行时先建立JAVA Project项目,再加入原代码即可以成功运行)

JAVA服务器、.NET客户端

.NET服务器、.JAVA客户端


JAVA与.NET的相互调用——利用JNBridge桥接模式实现远程通讯

JAVA与.NET的相互调用——TCP/IP套接字相互调用的基本架构(附原代码)

JAVA与.NET的相互调用——通过Web服务实现相互调用(附原代码)

对JAVA与.NET开发有兴趣的朋友欢迎加入QQ群:59557329 点击这里加入此群