已阅读5页,还剩25页未读, 继续免费阅读
版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
JAVA与.NET的相互调用TCP/IP相互调用基本架构TCP/IP套接字的概念TCP/IP(传输控制协议/网际协议)是网络互连的通信协议,通过它可以实现各种异构网络或异种机之间的互联通信。TCP/IP是Transmission Control Protocol/Internet Protocol的简写,中文译名为传输控制协议/因特网互联协议,又叫网络通讯协议,这个协议是Internet最基本的协议、Internet国际互联网络的基础,简单地说,就是由网络层的IP协议和传输层的TCP协议组成的。TCP/IP 定义了电子设备(比如计算机)如何连入因特网,以及数据如何在它们之间传输的标准。TCP/IP是一个四层的分层体系结构。高层为传输控制协议,它负责聚集信息或把文件拆分成更小的包。低层是网际协议,它处理每个包的地址部分,使这些包正确的到达目的地。 TCP/IP已成为当今计算机网络最成熟、应用最广的互联协议。Internet采用的就是 TCP/IP协议,网络上各种各样的计算机上只要安装了TCP/IP协议,它们之间就能相互通信。 TCP/IP套接字通讯的开发在众多的开发语言中,绝大部分的开发语言都支持TCP/IP协议通讯,开发过程也十分相像,先设置好Socket,然后由客户端发送请求信息,服务器连接客户端接收到请求后再返还信息。而在.NET系统当中则稍有不同,系统把Socket对象包装在TcpClient对象内,对Socket对象的生命周期进行管理。在开发过程当中,服务器与客户端的开发语言有所不同的情况经常发生,服务器是在JDK1.6的环境下进行开发的,客户却要求使用.NET开发客户端,这往往会令开发人员感到困惑!下面在下使用JAVA为服务器,.NET为客户端为例子,为大家介绍一下如何使用TCP/IP协议进行JAVA .NET之间的相互调用。像TCP/IP实现聊天室这样的例子很多,开发起来也比较简单,因为通讯双方都是使用String来传送信息。而在真正建立ERP、OA、CRM等系统的时候,通讯双方都必须先建立一套统一的通讯契约,才能实现TCP/IP通讯,下面将为大家介绍一个比较典型的企业信息通讯实例。信息传送方式因为.NET与JAVA各有不同的特性,双方不可能直接通过的序列化对象来传输信息,常用的信息交换方式有以下三种:1. 最笨拙也是最复杂的一种传息方式,就是直接使用“头文件说明+字段属性”的方式。 这是一个既原始又麻烦的通讯方式,因为每个契约都要以二进制的方式发送一个请求,就算是同一类契约,随着参数的不同,每个请求的长度也会发生改变。这样的传息方式虽然是麻烦,但在不同开发语言相互调用的时候却经常会看到,这可能是因为开发人员对两种开发语言未能完全熟悉,所以倒置使用这最原始最简单的开发方式。2. 使用XML的信息传送方式,这是最常见,使用最广的信息传递方式。在绝大多数的开发平台都会支持XML,所以XML在Web网络传讯过程中最为常见。但XML最大的一个缺点就是过于堪舆,耗费大量的传输流量。3. 对于XML的缺点,JSON应运而生而且发展迅速,JSON本是源于Javascript的,多数只用于B/S的页面开发,但随着技术的发展和多个开发语言的支持,现今到处都可以看JSON的身影。因为JSON既提供一套跨平台的通讯方式,也免去XML复杂特性,受到各类型开发人员的欢迎。服务器端开发 通讯契约 首先建立一套服务器与客户端同时接受通讯契约, Contract 的name特性是契约的名称,服务器会通过此名称在Contracts.xml文件中找到该契约,然后根据output的package属性,class属性,method属性找到该契约的包名称,类名,调用的方法等属性。 /name为契约名,服务器与客户端必须同时遵守此契约 获取Age等于此值的People对象集 /说明此契约内容 Manager /接收到GetPersonByAge请求时所调用的包名称 PersonManager /接收到GetPersonByAge请求时所调用的类名称 GetListByAge /接收到GetPersonByAge请求时所调用的处理方法名称 获取ID等于此值的People对象 Manager PersonManager GetListByID 以JSON方式实现信息传送 尽管目前在C/S的开发当中大部分还是使用序列化对象和分节字段的方式进行双方通讯,但在这个实例当中,在下想以JSON通讯方式为例子来实现。首先,客户端会使用额定格式的JSON向服务器发送请求: “ContractName”:“GetPeopleByAge”,“Params”:23ContractName代表着契约名称,系统会根据此名称在Contracts.xml文件中找到Name等于GetPeopleByAge的Contract项。然后在对应Output的子项Package,Class,Method中查找到对应的包,类型和方法。Params是客户端传输过来的参数,服务器端会调用对象的方法输入参数23后,得到计算结果,最后把结果返还到客户端。在这里有两点是值得注意的,第一点是JSON中的契约格式是固定的,服务器与客户端都必须遵守此契约,在ContractName中输入是必须对应的契约名称,而在Params中输入的必输是一个参数的集合,哪怕里面只包含有一个参数。第二点是在Contracts.xml文件,Output里面的 Package,Class,Method是服务器端自定义的,它只是绑定了服务器端实现GetPersonByAge契约的方法,而这些方法并不是固定,服务器可以根据系统的需要而修改。这个做法有点像Struts里面的Struts.xml文件,其意义就是在于使服务器的处理方法与客户端发送的请求实现分离。 基本结构 系统的基本结构如图,客户端会以JSON方式“ContractName”:“GetPeopleByAge”,“Params”:23发送请求到服务器,服务器会利用“数据转换层”把接收到的请求转换成Contract对象。然后逻辑转换层会根据该Contract对象调用对应的方法,最后把计算结果以JSON方式返回到客户端。 注意在服务器与客户端信息交换的过程中,都是使用JSON格式。 JSON数据转换 在服务器端,当接到到客户端请求后,Transfer类负责把接收到的JSON数据转换成Contract对象。在Transfer里面使用org.json工具包作为JSON的转化工具,org.json工具包可于以下网址下载/java/index.html。而Implement类包含GetResult(Contract contract )方法,其作就是根据contract对象Package,Class,Method等属性,调用“逻辑转换层”的对应方法,最后把计算结果返还给InputControl。服务器端接收请求后就会直接调用InputControl对输入的数据进行处理。/Contract实体类,包含契约的package,class,method,params等多个属性package Model;import org.json.JSONArray;public class Contract private String package1; private String class1; private String method; private JSONArray params; public void setPackage1(String package1) this.package1 = package1; public String getPackage1() return package1; public void setClass1(String class1) this.class1 = class1; public String getClass1() return class1; public void setMethod(String method) this.method = method; public String getMethod() return method; public void setParams(JSONArray params) this.params = params; public JSONArray getParams() return params; /把输入的String字符串转化为Contract对象/在这里使用org.json工具包作为JSON的转化工具,org.json工具包可于以下网址下载/java/index.htmlpackage Common;import java.io.File;import java.io.IOException;import javax.xml.parsers.DocumentBuilder;import javax.xml.parsers.DocumentBuilderFactory;import javax.xml.parsers.ParserConfigurationException;import Model.Contract;import org.json.*;import org.w3c.dom.Document;import org.w3c.dom.Element;import org.w3c.dom.NodeList;import org.xml.sax.SAXException;public class Transfer private Transfer() private static String contractName=null; private static Contract contract=new Contract(); private static JSONObject jsonObject=null; public static Contract GetContract(String data) throws Exception, JSONException, ParserConfigurationException, SAXException, IOException jsonObject=new JSONObject(data); /把字符串转化为JSONOject对象 GetContractName(); /获取契约名称 GetProperty(); /获取契约的package,class,method属性 GetParams(); /获取契约的参数集 return contract; /* * 获取契约对应的包名称,类名称,方法名称 */ private static void GetProperty() throws Exception File file = new File(Contracts.xml); DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); DocumentBuilder builder = factory.newDocumentBuilder(); Document doc = builder.parse(file); NodeList nodeList = doc.getElementsByTagName(Contract); Element contractElement=null; for (int i = 0; i nodeList.getLength(); i+) if(nodeList.item(i).getAttributes().item(0).getNodeValue().equals(contractName) contractElement=(Element)nodeList.item(i); break; if(contractElement!=null) Element outputElement=(Element)contractElement.getElementsByTagName(Output).item(0); contract.setPackage1(outputElement.getElementsByTagName(Package).item(0).getTextContent(); /获取包名称 contract.setClass1(outputElement.getElementsByTagName(Class).item(0).getTextContent(); /获取类名称 contract.setMethod(outputElement.getElementsByTagName(Method).item(0).getTextContent(); /获取方法名 else throw new Exception(未能找到对象的契约); /* * 获取契约名称 */ private static void GetContractName() throws JSONException contractName=jsonObject.getString(ContractName); /* * 获取输入参数 */ private static void GetParams() throws JSONException contract.setParams(jsonObject.getJSONArray(Params); /调用Contract对象里面包中的类的某个方法,获取计算结果package Common;import java.lang.reflect.Method;import java.util.HashMap;import java.util.Map;import org.json.JSONArray;import Model.*;public class Implement private Contract contract; private String fullName; private static Map objects=new HashMap(); /保存对象实体 private static Map classes=new HashMap(); /保存类名 /* * 先获取对应的对象,再用反射模式调用对象的方法,获取计算结果 */ public Object GetResult(Contract contract) this.contract=contract; this.fullName=contract.getPackage1()+.+contract.getClass1(); try Object manager=GetObject(); Class theClass=classes.get(fullName); Method method=theClass.getDeclaredMethod(contract.getMethod(),JSONArray.class); return method.invoke(manager, contract.getParams(); catch (Exception e) / TODO Auto-generated catch block e.printStackTrace(); return null; /* * 多次使用反射创建获取对象会损耗一定性能,所以此处使用单体模式获取对应的对象 */ private Object GetObject() throws InstantiationException, IllegalAccessException, ClassNotFoundException if(!objects.containsKey(fullName) Class theClass = Class.forName(fullName); classes.put(fullName,theClass); Object manager=theClass.newInstance(); objects.put(fullName, manager); return objects.get(fullName); /直接把接收到的二进制数据转换成String,然后通过Transfer把String转化为Contract对象,最后通过Implement获取运算结果package Common; import java.io.DataInputStream;import Model.Contract;public class InputControl private DataInputStream inputStream; public InputControl(DataInputStream inputStream) this.inputStream=inputStream; /* * 直接把接收到的二进制数据转换成String,然后通过Transfer把String转化为Contract对象,最后通过Implement对象获取运算结果 */ public Object GetResult() byte byteMessage=new byte1024; /在此处只获取测试数据,在真正运行时应使用分批缓存的方式 try int n=inputStream.read(byteMessage); String message=new String(byteMessage,ASCII); Contract contract=Transfer.GetContract(message); Implement implement=new Implement(); Object result=implement.GetResult(contract); return result; catch(Exception ex) ex.printStackTrace(); return null; 最后,系统通过OutputControl类把计算结果返还给客户端。/把计算结果返回到客户端package Common;import java.io.DataOutputStream;public class OutputControl private DataOutputStream outputStream; public OutputControl(DataOutputStream outputStream) this.outputStream=outputStream; public void Output(Object data) try outputStream.writeBytes(data.toString(); outputStream.flush(); catch(Exception ex) ex.printStackTrace(); /运行系统进行测试package Common;import java.io.*;import .*;public class Program private static ServerSocket serverSocket; public static void main(String args) throws ClassNotFoundException / TODO Auto-generated method stub Socket socket; try serverSocket=new ServerSocket(5100); /激活5100端口 while(true) /循环捕捉请求 socket=serverSocket.accept(); DataOutputStream outStream=new DataOutputStream(socket.getOutputStream(); /获取DataOutputStream输出流 DataInputStream inputStream=new DataInputStream(socket.getInputStream(); /获取DataInputStream输入流 /调用InputControl对象获取运算结果 InputControl inputControl=new InputControl(inputStream); Object result=inputControl.GetResult(); /调用OutputControl对象输入运算结果 OutputControl outputControl=new OutputControl(outStream); outputControl.Output(result); catch (Exception e) / TODO Auto-generated catch block e.printStackTrace(); 逻辑转换层 现在先开发一个例子作为参考,在完成客户端开发的时候就可以进行测试。这个例子是先在Manager包里面设置好一个类PersonManager,PersonManager类中包含一个名为GetListByAge的方法。在Contracts.xml文件设置一个名为GetPersonByAge的契约,客户端就可以通过这个契约在远程调用此方法获取计算结果。/设置Person对象package Model;public class Person private int id; private String name; private int age; public void setId(int id) this.id = id; public int getId() return id; public void setName(String name) = name; public String getName() return name; public void setAge(int age) this.age = age; public int getAge() return age; /开发PersonManagerpackage Manager;import java.util.ArrayList;import java.util.List;import org.json.JSONArray;import org.json.JSONException;import Model.*;public class PersonManager /* * 测试数据 */ private List GetList() List personList=new ArrayList(); Person person1=new Person(); person1.setId(0); person1.setAge(23); person1.setName(Mike); personList.add(person1); Person person2=new Person(); person2.setId(1); person2.setAge(29); person2.setName(Leslie); personList.add(person2); Person person3=new Person(); person3.setId(2); person3.setAge(21); person3.setName(Jack); personList.add(person3); Person person4=new Person(); person4.setId(3); person4.setAge(23); person4.setName(Rose); personList.add(person4); return personList; /* * 获取年龄等于age参数的Person,因为数据将返还给客户端,所以这时把输出数据转化为JSONArray */ public JSONArray GetListByAge(JSONArray jsonList) throws JSONException int age=jsonList.getInt(0); /因为输入参数为一个集合params,所以即使只包括一个参数,也是通过要jsonList的第一个参数来获取的。 List personList=GetList(); List resultList=new ArrayList(); for(int n=0;npersonList.size();n+) if(personList.get(n).getAge()=age) resultList.add(personList.get(n); JSONArray jsonArray=new JSONArray(resultList); return jsonArray; 然后在Contracts.xml设置绑定: /契约名称 获取Age等于此值的People对象集 /文字说明 Manager /绑定包 PersonManager /绑定类 GetListByAge /绑定处理方法绑定以后,在完成客户端开发的时候就可以进行测试。使用这开发模式的好处在于利用JSON作用数据传输的桥梁,解决不同开发平台之间数据难以同步的问题。使用JSON比XML更容易操作,可以减少传输流量,而且受到各开发语言的支持。使用Contracts.xml在服务器绑定处理方式,使服务器的处理方法与客户端发送的请求实现分离。下面开始介绍一下客户端的开发。客户端开发客户端的开发的开发相对简单,因为契约是使用 “ContractName”:“GetPeopleByAge”,“Params”:23 JSON方式进行传送,所以先开发一个MessageEntity实体类来承载契约。namespace Model DataContract public class MessageEntity /契约名称 DataMember public string ContractName get; set; /注意参数使用集合的方式来传送 DataMember public IList Params get; set; 然后开发一个MessageManager信息管理器来管理契约的传送过程,因为Framework4.0里面,未能对JSON数据中集合的转换提供一个简单函数,所以在MessageManager里面使用了一个Newtonsoft.Json工具包,该工具包里面对JSON的操作有着强大支持,可以在/官方网站下载:using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Net.Sockets;using System.Runtime.Serialization.Json;using System.IO;using System.Threading;using Model;using Newtonsoft.Json;namespace Common public class MessageManager private static TcpClient _tcpClient; /设置tcpClient对象 public static TcpClient TcpClient set _tcpClient = value; /此处只使用静态方法实现数据传送,发送请求后使用Thread.Sleep等待运算结果,这样存在一定风险,也会降低效率 /在大型的开发当中应该进一步改善,把信息发送与信息接收分开处理 public static object GetMessage(MessageEntity message, Type type) NetworkStream networkStream = _tcpClient.GetStream(); /利用DataContractJsonSerializer将MessageEntity对象实现序列化,发送到服务器 DataContractJsonSerializer jsonSerializer = new DataContractJsonSerializer(typeof(MessageEntity); lock (networkStream) jsonSerializer.WriteObject(networkStream, message); networkStream.Flush(); Thread.Sleep(500); /获取回传信息,这里设置接收值1024个字节 /在实际的开发当中应该使用分批缓存的方式实现数据接收 byte messageByte = new byte1024; int n = 0; lock (networkStream) n = networkStream.Read(messageByte, 0, 1024); if (n = 0) return null; /根据输入的type对象,把二进制信息转化为对应的对象 string jsonMessage = Encoding.ASCII.GetString(messageByte); /利用Netonsoft.Json工具集将获取的JSON数据转化对象 object returnValue = JavaScriptConvert.DeserializeObject(jsonMessage, type); return returnValue; 下面开发一个GetPersonByAge 契约作为例子:using System;using System.Collections.Generic;using System.Text;using Model;using Common;namespace DAL public class PersonDAL / / 建立MessageEntity对象,注意输入额定契约名称及数据参数,获取查询结果 / / Person的年龄 / 获取年龄等于此值的Person对象集 public IList GetPersonByAge(int age) /先建立一个MessageEntity对象,设定其ContractName及参数集合 /注意ContractName的值必须与服务器端Contracts.xml文件中Contract 项的 name 特性相对应 MessageEntity messageEntity = new MessageEntity(); messageEntity.ContractName = GetPersonByAge; messageEntity.Params = new List age ; /调用MessageManager的GetMessage方法获取计算结果 IList personList = (List)MessageManager.GetMessage(messageEntity, typeof(List); return personList; PersonDAL类中的GetPersonByAge方法就
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 2025年环保竞赛试卷模板及答案
- 燃气轮机运行值班员岗前工作实操考核试卷含答案
- 翻车机工安全生产知识模拟考核试卷含答案
- 砌筑工安全生产能力考核试卷含答案
- 蒙药材种植员操作评估水平考核试卷含答案
- 开清棉工岗前实操知识水平考核试卷含答案
- 网商岗前节能考核试卷含答案
- 区域地质调查员岗前进度管理考核试卷含答案
- 中央空调系统运行操作员改进测试考核试卷含答案
- 2025年宠物智能代理市场潜力与商业模式创新报告
- 通信工程生涯发展展示
- 中国融通集团在线考试题
- 陕09J05 卫生间、盥洗室及洗池图集
- 塑料制品装配操作工培训教材
- 第5章齿轮传动
- 居住人口登记表
- 钳工实操试卷-共44套
- 黑山羊养殖技术培训课件
- (完整版)二手房交房确认书
- 07FK02防空地下室通风设备安装图集
- 6人小品《没有学习的人不伤心》台词完整版
评论
0/150
提交评论