博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
.Net remoting, Webservice,WCF,Socket区别
阅读量:6572 次
发布时间:2019-06-24

本文共 13679 字,大约阅读时间需要 45 分钟。

传统上,我们把计算机后台程序(Daemon)提供的功能,称为"服务"(service)。比如,让一个杀毒软件在后台运行,它会自动监控系统,那么这种自动监控就是一个"服务"。
通俗地说,"服务"就是计算机可以提供的某一种功能。
根据来源的不同,"服务"又可以分成两种:一种是"本地服务"(使用同一台机器提供的服务,不需要网络),另一种是"网络服务"(使用另一台计算机提供的服务,必须通过
网络才能完成)。"网络服务"(Web Service)的本质,就是通过网络调用其他网站的资源。举例来说,去年公开课的时候我在ArcGIS Engine中调用过一个天气预报的信息,并将信息和具体的位置关联起来。所以,Web service让你的网站可以使用其他网站的资源,比如在网页上显示天气、地图、twitter上的最新动态等等。
综上:WebService是两个计算机之间通讯(交谈)的技术。并且现在炒的很火的SOA、云计算在技术层面上都是WebService。
除了WebService 通讯外,系统间通讯有很多种技术,如像qq的这种Socket通讯在银行系统中广泛应用,.Net Remoting、DCom等通讯方式也应用很 多,
但是这些方式有如下缺点:
•        通讯数据格式不统一,同样一个"你好"这样的字符串在不同的协议中有不同的表示方法,异构系统集成很麻烦。一个系统一个模样。
•        采用Socket、 .Net Remoting、DCom需要打开很多端口,而企业网络安全的一个基本原则就是“尽可能少的打开端口”,很多企业网络甚至严格规定“只能打开80端口”,
因此需要一种“跨防火墙”的技术(跨防火墙就是走80端口进行通讯)
•        这些通讯方式的协议是不开放的,要想知道服务提供了哪些方法、如何调用,必须能够自描述
 
WSDL(Web Service Description Language)

你会怎样向别人介绍你的Web service有什么功能,以及每个函数调用时的参数呢?你可能会自己写一套文档,你甚至可能会口头上告诉需要使用你的Web service的人。

这些非正式的方法至少都有一个严重的问题:当程序员坐到电脑前,想要使用你的Web service的时候,他们的工具(如Visual Studio)无法给他们提供任何帮助,因为这些工具根本就不了解你的Web service。解决方法是:用机器能阅读的方式提供一个正式的描述文档。Web service描述语言(WSDL)就是这样一个基于XML的语言,用于描述Web service及其函数、参数和返回值。因为是基于XML的,所以WSDL既是机器可阅读的,又是人可阅读的,这将是一个很大的好处。一些最新的开发工具既能根据你的Web service生成WSDL文档,又能导入WSDL文档,生成调用相应Web service的代码

Web服务器描述语言是用XML文档来描述Web服务的标准,是Web服务的接口定义语言,由Ariba、Intel、IBM、MS等共同提出,通过WSDL,可描述Web服务的三个基本属性:
·服务做些什么——服务所提供的操作(方法)
·如何访问服务——和服务交互的数据格式以及必要协议
·服务位于何处——协议相关的地址,如URL

 

XML和XSD 

可扩展的标记语言(XML)是Web service平台中表示数据的基本格式。除了易于建立和易于分析外,XML主要的优点在于它既是平台无关的,又是厂商无关的。无关性是比技术优越性更重要的:软件厂商是不会选择一个由竞争对手所发明的技术的。 
XML解决了数据表示的问题,但它没有定义一套标准的数据类型,更没有说怎么去扩展这套数据类型。例如,整形数到底代表什么?16位,32位,还是64位?这些细节对实现互操作性都是很重要的。W3C制定的XML Schema(XSD)就是专门解决这个问题的一套标准。它定义了一套标准的数据类型,并给出了一种语言来扩展这套数据类型。Web service平台就是用XSD来作为其数据类型系统的。当你用某种语言(如VB.NET或C#)来构造一个Web service时,为了符合Web service标准,所有你使用的数据类型都必须被转换为XSD类型。你用的工具可能已经自动帮你完成了这个转换,但你很可能会根据你的需要修改一下转换过程。在第二章中,我们将深入XSD,学习怎样转换自定义的数据类型(例如类)到XSD的类型。 
SOAP 
Web service建好以后,你或者其他人就会去调用它。简单对象访问协议(SOAP)提供了标准的RPC方法来调用Web service。实际上,SOAP在这里有点用词不当:它意味着下面的Web service是以对象的方式表示的,但事实并不一定如此:你完全可以把你的Web service写成一系列的C函数,并仍然使用SOAP进行调用。SOAP规范定义了SOAP消息的格式,以及怎样通过HTTP协议来使用SOAP。SOAP也是基于XML和XSD的,XML是SOAP的数据编码方式。第三章我们会讨论SOAP,并结识SOAP消息的各种元素。 
综上我们还可以得出以下结论:

请求、返回的XML数据格式(有哪些节点、节点的名字等等)WebService 用SOAP协议进行规定,方法描述信息XML用WSDL协议规定。WebService技术是与语言、平台无关,因此.net可以访问java编写的WebService、java也可以访问.net编写的webservice,php、python等各种语言也几乎都支持webservice,因此可以说webservice可以实现跨语言方法调用。

但是如果自己构建请求、返回xml,解析xml请求,自己负责方法描述信息更新是很麻烦的,.net就提供了简化开发WebService。 

WebService的创建和使用

WebService的使用过程包括服务器端代码的编写和客户端代码的调用,服务器端的代码比较容易,只是在使用的时候需要添加一些[WebService]和[WebMethod]这样的标记

 1、  服务器端(ServerWeb):就想写普通方法一样,不需要处理请求、响应。服务器端新建“Web服务”(asmx),在远端可以调用的方法上标注[WebMethod]。

 

using System;   using System.Linq;   using System.Web;   using System.Web.Services;   using System.Web.Services.Protocols;   using System.Xml.Linq;   [WebService(Namespace = "http://tempuri.org/")]   [WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]   // 若要允许使用 ASP.NET AJAX 从脚本中调用此 Web 服务,请取消对下行的注释。   // [System.Web.Script.Services.ScriptService]   public class Service : System.Web.Services.WebService   {       public Service () {           //如果使用设计的组件,请取消注释以下行            //InitializeComponent();        }       //[WebMethod]       //public string HelloWorld()       //{       //    return "Hello World";       //}       [WebMethod(Description = "求和的方法")]       public double addition(double i, double j)       {           return i + j;       }       [WebMethod(Description = "求差的方法")]       public double subtract(double i, double j)       {           return i - j;       }       [WebMethod(Description = "求积的方法")]       public double multiplication(double i, double j)       {           return i * j;       }       [WebMethod(Description = "求商的方法")]       public double division(double i, double j)       {           if (j != 0)               return i / j;           else              return 0;       }   }

 

 

 2、客户端添加对asmx的“服务器引用”,然后就可以调用***SoapClient类中的方法。就“好像”直接调用了服务端的方法(因为VS2010中有两种添加服务的方法,一种是早期的针对net2.0的,如果是2.0的话,会生成一个跟服务端同类名的类)。

 

添加服务引用的时候工具读取asmx的WSDL自动生成了ServiceReference1中的类,这些类帮我们来拼Http请求,并且把Http返回值拆成函数的返回。

客户端“添加服务引用”,填写asmx的地址。然后就可以调用Service References下自动生成的***SoapClient类了。

用WebService的时候如果服务端的接口定义发生变化,则需要重新添加对服务端的引用,因为Service References中的类是工具读取WSDL定义自动生成的。在服务引用上点击右键,选择“更新服务引用”。如果只是修改了WebService内部实现,而接口没变,则不需要“更新服务引用”,因为WSDL没变,Soap没变。    

static void Main(string[] args)        {            ServiceTest.ServiceSoapClient stsoap = new ServiceTest.ServiceSoapClient();           double c= stsoap.addition(1, 2);           Console.WriteLine(c.ToString());            // vs 2010中两种方式            ServiceTest.Service st = new ServiceTest.Service();            double a = st.addition(1, 2);            Console.WriteLine(a.ToString());            Console.ReadKey();        }

 

 

 

.Net remoting

 

Net remoting 是简化网络通讯的技术,底层仍然是TCP等东西。

remoting要添加对System.Runtime.Remoting的引用

编写服务接口类库项目,正常写法!WebService中WSDL相当于对服务端方法的描述;.net Remoting中走的是二进制数据,因此必须一个描述服务端方法的接口类库。这个类库里面往往是一个接口,定义了服务器端实现方法的方法名,客户端和服务器端都需要引进用这个接口。

服务端实现服务接口,继承自MarshalByRefObject,然后运行备注中的代码注册服务。

客户端代码在备注中

Remoting和WebService的区别:Remoting效率高,走的是普通TCP, WebService则是Http协议,需要IIS、ASP.Net、XML解析等,效率低。Remoting适合于内网通讯, WebService适合于外网通讯。

除非项目要求,否则以后尽量用WCF。.Net Remoting是微软私有协议,因此如果要跨平台调用还是普通Socket或者WebService。

用法说明:

 1、新建接口项目,定义服务接口。

注意:remoting要添加对System.Runtime.Remoting的引用

 

using System;using System.Collections.Generic;using System.Linq;using System.Text;namespace NETRemotiongInterface{    public interface IRemotingTest    {        double add(double a,double b);    }}

 

 

 

2、新建服务器端项目(控制台的,或者WinForm,或者Windows服务等)

定义实现服务接口的类,还要继承继承自MarshalByRefObject类

 

using System;using System.Collections.Generic;using System.Linq;using System.Text;using NETRemotiongInterface;namespace RemotingServer{    class TestServiceImp : MarshalByRefObject,IRemotingTest
{ #region IRemotingTest 成员 public double add(double a, double b) { return a + b; } #endregion } }

 

 

 

3,服务器启动时调用

 

using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Runtime.Remoting.Channels.Tcp;using System.Runtime.Remoting;using System.Runtime.Remoting.Channels;namespace RemotingServer{    class Program    {        static void Main(string[] args)        {             TcpChannel tcpChannel = new TcpChannel(9999);            ChannelServices.RegisterChannel(tcpChannel);             //注册服务:第一个参数为服务的实现类,第二个参数为服务的名字。            RemotingConfiguration.RegisterWellKnownServiceType(typeof(TestServiceImp),                "test", WellKnownObjectMode.Singleton);    //注册服务。如果控制台程序,控制不要让程序退出,            //主要目的是不要让服务器退出            while (true)            {                string s = Console.ReadLine();                if (s == "quit")                {                    return;                }            }        }    }}
 

,4、客户端:新建客户端项目,引用服务接口

 

namespace RemotingClient{    class Program    {        static void Main(string[] args)        {            TcpChannel tcpChannel = new TcpChannel();            ChannelServices.RegisterChannel(tcpChannel, false);            IRemotingTest test =                (IRemotingTest)Activator.GetObject(typeof(IRemotingTest), "tcp://127.0.0.1:9999/test");//第一个参数为服务实现的接口,第二个参数为服务的地址:最后一部分是服务在服务器端RegisterWellKnownServiceType时第二个参数的名字然后就可以调用服务端方法了。            double a = test.add(1, 2);            Console.WriteLine(a.ToString());            Console.ReadKey();         }    }}

 

从WebService和NetRemoting来看客户端都要知道服务器端暴露的功能,这个是必须的,要不然客户端怎么调用,WS是通过一个地址,而NetRemoting是通过一个接口类库,当然这也只是表面现象。

                                                                 WCF

WCF(Windows Communication foundation)是微软的统一网络通讯开发的技术,无论底层用.Net Remoting还是WebService还是Restful还是MSMQ等,只要修改配置文件即可。所以WCF并不是新技术。

WCF和.Net Remoting、WebService等技术的关系就像ADO.Net和SQLServer、Oracle驱动的关系一样。通过VS的“WCF服务配置编辑器”简化配置,修改不同的协议。

一开始内网运行就行,后来想运行到公网,那么如果一开始用.net remoting写后来改成WebService还是有工作量的,因为写法不一样,但是用WCF就不一样了。

新建“WCF服务库”,WCF服务库可以Host在IIS上、单独的WinForm程序等。

WCF、.Net Remoting和WebService的关系: .Net Remoting是普通的TCP通讯,适合于局域网,效率高; WebService是基于Http协议,适合于广域网,效率低;WCF是对.Net Remoting、 WebService等的简化、统一,可以通过配置来切换不同的底层实现,代码几乎不用动。

 

关于WCF的例子,我也懒得写了,给出一篇博文,大家可以自行研究:

 

Socket的例子我们可能见得比较多,这里就不罗嗦了,啰嗦了,直接上代码:

 

服务器端

 

using System;using System.Net;using System.Net.Sockets;using System.Text;public class SynchronousSocketClient {    public static void StartClient() {        // Data buffer for incoming data.        byte[] bytes = new byte[1024];        // Connect to a remote device.        try {            // Establish the remote endpoint for the socket.            // This example uses port 11000 on the local computer.            IPHostEntry ipHostInfo = Dns.Resolve(Dns.GetHostName())            IPAddress ipAddress = ipHostInfo.AddressList[0];            IPEndPoint remoteEP = new IPEndPoint(ipAddress,11000);            // Create a TCP/IP  socket.            Socket sender = new Socket(AddressFamily.InterNetwork,                 SocketType.Stream, ProtocolType.Tcp );            // Connect the socket to the remote endpoint. Catch any errors.            try {                sender.Connect(remoteEP);                Console.WriteLine("Socket connected to {0}",                    sender.RemoteEndPoint.ToString());                // Encode the data string into a byte array.                byte[] msg = Encoding.ASCII.GetBytes("This is a test
"); // Send the data through the socket. int bytesSent = sender.Send(msg); // Receive the response from the remote device. int bytesRec = sender.Receive(bytes); Console.WriteLine("Echoed test = {0}", Encoding.ASCII.GetString(bytes,0,bytesRec)); // Release the socket. sender.Shutdown(SocketShutdown.Both); sender.Close(); } catch (ArgumentNullException ane) { Console.WriteLine("ArgumentNullException : {0}",ane.ToString()); } catch (SocketException se) { Console.WriteLine("SocketException : {0}",se.ToString()); } catch (Exception e) { Console.WriteLine("Unexpected exception : {0}", e.ToString()); } } catch (Exception e) { Console.WriteLine( e.ToString()); } } public static int Main(String[] args) { StartClient(); return 0; }}

 

客户端

using System;using System.Net;using System.Net.Sockets;using System.Text;public class SynchronousSocketListener {        // Incoming data from the client.    public static string data = null;    public static void StartListening() {        // Data buffer for incoming data.        byte[] bytes = new Byte[1024];        // Establish the local endpoint for the socket.        // Dns.GetHostName returns the name of the         // host running the application.        IPHostEntry ipHostInfo = Dns.Resolve(Dns.GetHostName());        IPAddress ipAddress = ipHostInfo.AddressList[0];        IPEndPoint localEndPoint = new IPEndPoint(ipAddress, 11000);        // Create a TCP/IP socket.        Socket listener = new Socket(AddressFamily.InterNetwork,            SocketType.Stream, ProtocolType.Tcp );        // Bind the socket to the local endpoint and         // listen for incoming connections.        try {            listener.Bind(localEndPoint);            listener.Listen(10);            // Start listening for connections.            while (true) {                Console.WriteLine("Waiting for a connection");                // Program is suspended while waiting for an incoming connection.                Socket handler = listener.Accept();                data = null;                // An incoming connection needs to be processed.                while (true) {                    bytes = new byte[1024];                    int bytesRec = handler.Receive(bytes);                    data += Encoding.ASCII.GetString(bytes,0,bytesRec);                    if (data.IndexOf("
") > -1) { break; } } // Show the data on the console. Console.WriteLine( "Text received : {0}", data); // Echo the data back to the client. byte[] msg = Encoding.ASCII.GetBytes(data); handler.Send(msg); handler.Shutdown(SocketShutdown.Both); handler.Close(); } } catch (Exception e) { Console.WriteLine(e.ToString()); } Console.WriteLine("\nPress ENTER to continue"); Console.Read(); } public static int Main(String[] args) { StartListening(); return 0; }}

 

 

 

 

来点这四个的总结的话。

 

1:socket VS remoting

 

    使用socket无疑是效率最高的。但是,在复杂的接口环境下,socket的开发效率也是最低的。故在兼顾开发效率的情况下,可以使用remoting来代替socket开发。并且:

    1、Tcp通道的Remoting速度非常快。
    你可以通过端口查看工具,发现remoting比直接socket传输的内容,应该是属于同一个数量级的。我的另一个担心是,大客户端数量的情况下,remoting传输效率会不会很低,结果经过现场测试,同时对300个客户端进行数据通信,不存在信息丢失情况。
    2、虽然是远程的,但是非常接近于本地调用对象。
    也就是完全符合面向对象思想。
    3、可以做到保持对象的状态
    直接使用socket传输机制,我们必须花大量的精力来处理异常、断网、死机等现象,使用remoting,这些工作会大大简化。

 

2:remoting vs webservice 

    1、webservice在framework2.0状态下只能寄宿于IIS等应用服务器中。微软直到3.0才提供了servicehost来寄宿 webservice,这就极大地限制了webservice在使用中的灵活性。在framework2.0环境下,如果你有一个应用要脱离IIS而存 在,就不得不抛弃webservice。(除非你想代码实现一个WEB应用服务器)
    2、remoting可寄宿在你自己的代码中,也可寄宿在windows服务及IIS中。最大程度的提供了开发和部署的灵活性。
    3、remoting在使用http通道的时候,也如webservice一样支持穿透路由。
    4、remoting与websercie相比,提供双向通信。哪怕是将remoting寄宿在IIS中,也支持。
    5、webservice客户端自动生成的代理类比较复杂。而remoting一般来说,都是手动编写客户端代码。
    6、当然,webservice最主要优势是,它是一个行业标准,而remoting只是微软自己内部的标准,如果你的应用要脱离微软的平台,就只能使用webservice了。

    7,NetRemoting的优点是用户既可以使用TCP信道方式进行二进制流方式通信,也可以使用HTTP信道进行SOAP格式的性通信,而WebService只能使用HTTP通道

    8, NetRemotiong效率相对WebService要高不少;

    9, Netremoting可以用于有状态的情况,而WebService只能使用无状态的情况。

 

3:remoting vs wcf

    与wcf的比较,更多的是从平台的普及度上来说。在当前环境下,2.0的普及度还是最高的。如果哪一天3.0甚至4.0普及了,当然WCF是最好的。

参考:

        

        

       http://blog.csdn.net/huangxinfeng/article/details/4967629

 

 

你可能感兴趣的文章
linux下bus、devices和platform的基础模型 【转】
查看>>
如何优雅的使用RabbitMQ
查看>>
js json 对象相互转换
查看>>
jQuery中click事件多次触发解决方案
查看>>
java IO
查看>>
css3中定义required,focus,valid和invalid样式
查看>>
Spark history-server 配置 !运维人员的强大工具
查看>>
Atitit.http httpclient实践java c# .net php attilax总结
查看>>
Atitit.识别损坏的图像
查看>>
swift获取图片像素颜色值
查看>>
MyCat:取代Cobar数据库中间件
查看>>
ajax提交复杂对象数据
查看>>
wordpress发送测试邮件
查看>>
用PyAIML开发简单的对话机器人
查看>>
Android 7.1 App Shortcuts使用
查看>>
解决: is not found. Have you run APT to generate them?
查看>>
jenkins配置记录(1)--添加用户权限
查看>>
Android bitmap绘制文字自动换行
查看>>
express下使用ES6
查看>>
django中的filter和get的区别 (MultipleObjectsReturned: get() returned more than one Publisher --)(DoesN...
查看>>