学习webservice的相关文章-1

  •  Web Service的传输协议概述
  •  ASP.NET 中,XML Web Service支持3种协议来与用户交流数据。这3种协议分别是:
    1HTTP-GET
    2HTTP-POST
    3SOAP
    在这3种协议中,HTTP(Hypertext Transfer Protocol)已经是众所周知的协议了,它是XML Web Service数据传输的标准,其中包括使用SOAP传输数据。HTTPSOAP消息压缩,然后以HTTP传输协议的形式进行网络传输。然而在XML Web Service下使用HTTP-GETHTTP-POST的时候,事实上是指有关单独使用HTTP调用XML Web Service中的方法的能力,而不使用SOAP
    HTTP中,GETPOST并不是一种协议,它们是可以用来与Web Service交互的几种方法中的其中两种。然而,这二种方法的传送参数和数据的能力使它们变成了一种简单的,非常适合用来调用XML Web Service的工具。
    SOAPXML Web Service最常用到的连接协议。与HTTP相比,SOAP显的更为复杂,但却拥有更强的接受能力。SOAP是一种以XML为基础的协议,它提供一种将数据打包(Packaging)和编码(Encoding)的方法,以用于网络的数据传输。任意一个用户都可以使用SOAP协议与任何一个XML Web Service进行通信,甚至于这个XML Web Service不是建立在.NET 平台上的,比如说Java的,都可以利用SOAP来进行数据传输。因此可见,SOAP也是Language Independent(语言独立性)
    2
    HTTP-GET
    HTTP-POSTHTTP-GET HTTP-POST 是使用 HTTP(超文本传输协议)谓词以及与之关联的请求语义将参数作为名称/值对编码和传递的标准协议。每个协议都由一系列 HTTP 请求头组成,这些头与一些其他信息一起定义客户端向服务器请求的内容,而在成功时,服务器将用一系列 HTTP 响应头和所请求的数据响应。
    HTTP-GET的处理特征如下:
    l
    将数据添加到URL
    l
    利用一个问号(“?”)代表URL地址的结尾与数据的开端。
    l
    每一个数据的元素以 名称/值的形式出现。
    l
    利用一个分号(“;”)来区分多个数据元素。
    HTTP-POST的处理特征如下:
    l
    将数据包括在HTTP主体中。
    l
    同样的,数据的元素以名称/值的形式出现。
    l
    但是每一个数据元素分别占用主体的一行。
    3
    SOAP
    协议SOAPSimple Object Access Protocol)简单对象访问协议,它是轻型协议,用于分散的、分布式计算环境中交换信息。SOAP有助于以独立于平台的方式访问对象、服务和服务器。它借助于XML,提供了HTTP所需的扩展。
    SOAP协议规范由4个主要的部分组成。
    第一部分:SOAP封装(Envelop)定义了一个描述消息的内容多少、谁发送、谁应当接受并且处理以及如何处理它们的框架。
    第二部分:SOAP编码规则(Encoding Rules)定义了可选数据编码规则,用于表示应用程序定义的数据类型和直接图表,以及一个用于序列化非语法数据模型统一标准。
    第三部分:SOAP RPC表示(RPC Representation)定义一个远程调用风格(请求/响应)信息交换的模式。
    第四部分:SOAP绑定(Binding)定义了SOAPHTTP之间的绑定和使用底层协议的交换。
    SOAP协议可以简单地理解为:SOAP=RPC+HTTP+XML,即采用HTTP作为通信协议,RPC作为一致性的调用途径,XML作为数据传送的格式,从而允许服务提供者和服务客户经过防火墙在Internet上进行通信交互。

  • WebService传输数据流及数据交互详解

    为了实现一个嵌入式设备和一个host在IIS server上面的WebService交互,使用了MF3.0里面的DPWS方法。不过这个东西实在是Hard to use。今天就记录下使用的过程中的第一步,首先分析一个Client和一个Webservice是如何交互的,发送的数据流的格式,然后在Device里面封装一下Soap头用httpClient发送出去。

    这里用到了WireShark来分析网络交互的数据包。

    首先定义一个运行在IIS上面的简单的WebService:

    namespace Cashfree.Vending.Web.WebService
    {
        [WebService(Namespace = "http://tempuri.org/")]
        [WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
        [System.ComponentModel.ToolboxItem(false)]
        public class IISWS : System.Web.Services.WebService
        {

            [WebMethod]
            public string HelloWorld()
            {
                return "Hello Robin";
            }

            [WebMethod]
            public int add(int a, int b)
            {
                return a + b;
            }
        }
    }

    然后定义一个Client来访问这个Webservice:

    namespace Cashfree.Vending.IISWSInvoke
    {
        class Program
        {
            static void Main(string[] args)
            {
                ISWSSoapClient isClient = new ISWSSoapClient();           
                Console.WriteLine(isClient.add(2, 3));
            }
        }
    }

     

    接着,发布好Webservice之后,编译Client文件,然后找到Client的exe文件。

    这个时候打开WireShark,监视网卡上面发送的数据流,设置过滤,格式如下:

    (ip.addr eq 10.10.20.33 and ip.addr eq 192.168.0.100) and (tcp.port eq 1795 and tcp.port eq 8088)

    双击Client端访问IIS上面的Webservice,然后得到下面的一个数据包列表:

     

    这就得到了一个完整的交互过程,从下往上,最左边的一列是frame id。

    我把着几个frame,199到122解析出来得到了下面的格式:

     

    首先是客户端发送一个请求到iis的服务器上面,请求的地址是:/WebService/IISWS.asmx,这里是支持http1.1的。

    然后连接类型是一直保持连接。

     

     

    这是Server回的一个ACK到Client来保持连接。同时标识连接通了。

     

    然后Client端开始发送请求调用IIS上面的服务了,传递了两个参数:

    a=2,b=3;

    封装之后,有一个头部和一个body。Envelope在MF2.5的时候,有类库直接对应SOAP的头和Body。3.0的时候找XMLDocument找了N久没找到,MDSN上面明明写的是有的。但是这个地方如果使用MF来解析的话,可以调用XMLReader来实现。只不过比较麻烦,只是提供了最基础的XML访问的方法,需要一个节点一个节点的找。

     

    服务器端首先回一个ACK,然后跟着一个XMl文件表示调用的结果:

    <addResult>5</addResult>

    这里,俺们知道了传输的数据格式之后,就可以把Soap message封装好了之后直接传输给Server了。

    作者:lbq1221119

         通常我们在程序中需要调用WebService时,都是通过“添加Web引用”,让VS.NET环境来为我们生成服务代理,然后调用对应的Web服务。这样是使工作简单了,但是却和提供Web服务的URL、方法名、参数绑定在一起了,这是VS.NET自动为我们生成Web服务代理的限制。如果哪一天发布Web服务的URL改变了,则我们需要重新让VS.NET生成代理,并重新编译。在某些情况下,这可能是不能忍受的,我们需要动态调用WebService的能力。比如我们可以把Web服务的URL保存在配置文件中,这样,当服务URL改变时,只需要修改配置文件就可以了。
         说了这么多,实际上我们要实现这样的功能:

    public static object InvokeWebService(string url,  string methodname, object[] args)

         其中,url是Web服务的地址,methodname是要调用服务方法名,args是要调用Web服务所需的参数,返回值就是web服务返回的结果了。

         要实现这样的功能,你需要这几个方面的技能:反射、CodeDom、编程使用C#编译器、WebService。在了解这些知识后,就可以容易的实现web服务的动态调用了:

             #region InvokeWebService
            
    //动态调用web服务
            public static object InvokeWebService(string url, string methodname, object[] args)
             {
                
    return WebServiceHelper.InvokeWebService(url ,null ,methodname ,args) ;
             }

            
    public static object InvokeWebService(string url,  string classname, string methodname, object[] args)
             {
                
    string @namespace = "EnterpriseServerBase.WebService.DynamicWebCalling" ;
                
    if((classname == null) ||(classname == ""))
                 {
                     classname
    = WebServiceHelper.GetWsClassName(url) ;
                 }

                
    try
                 {
                    
    //获取WSDL
                     WebClient wc                   = new WebClient();
                     Stream stream                  
    = wc.OpenRead(url+"?WSDL");
                     ServiceDescription sd          
    = ServiceDescription.Read(stream);
                     ServiceDescriptionImporter sdi
    = new ServiceDescriptionImporter();
                     sdi.AddServiceDescription(sd,
    "","");
                     CodeNamespace cn                
    = new CodeNamespace(@namespace);
                    
                    
    //生成客户端代理类代码
                     CodeCompileUnit ccu             = new CodeCompileUnit();
                     ccu.Namespaces.Add(cn);
                     sdi.Import(cn ,ccu);
                     CSharpCodeProvider csc          
    = new CSharpCodeProvider();
                     ICodeCompiler icc               
    = csc.CreateCompiler();
                    
                    
    //设定编译参数
                     CompilerParameters cplist       = new CompilerParameters();
                     cplist.GenerateExecutable       
    = false;
                     cplist.GenerateInMemory         
    = true;
                     cplist.ReferencedAssemblies.Add(
    "System.dll");
                     cplist.ReferencedAssemblies.Add(
    "System.XML.dll");
                     cplist.ReferencedAssemblies.Add(
    "System.Web.Services.dll");
                     cplist.ReferencedAssemblies.Add(
    "System.Data.dll");

                    
    //编译代理类
                     CompilerResults cr = icc.CompileAssemblyFromDom(cplist, ccu);
                    
    if(true == cr.Errors.HasErrors)
                     {
                         System.Text.StringBuilder sb
    = new System.Text.StringBuilder();
                        
    foreach(System.CodeDom.Compiler.CompilerError ce in cr.Errors)
                         {
                             sb.Append(ce.ToString());
                             sb.Append(System.Environment.NewLine);
                         }
                        
    throw new Exception(sb.ToString());
                     }

                    
    //生成代理实例,并调用方法
                     System.Reflection.Assembly assembly = cr.CompiledAssembly;
                     Type t
    = assembly.GetType(@namespace+"."+classname,true,true);
                    
    object obj = Activator.CreateInstance(t);
                     System.Reflection.MethodInfo mi
    = t.GetMethod(methodname);

                    
    return mi.Invoke(obj,args);
                 }
                
    catch(Exception ex)
                 {
                    
    throw new Exception(ex.InnerException.Message,new Exception(ex.InnerException.StackTrace));
                 }
             }

            
    private static string GetWsClassName(string wsUrl)
             {
                
    string[] parts = wsUrl.Split('/') ;
                
    string[] pps   = parts[parts.Length-1].Split('.') ;

                
    return pps[0] ;
             }
            
    #endregion


         上面的注释已经很好的说明了各代码段的功能,下面给个例子看看,这个例子是通过访问http://www.webservicex.net/globalweather.asmx 服务来获取各大城市的天气状况。

                 string url = "http://www.webservicex.net/globalweather.asmx" ;
                
    string[] args = new string[2] ;
                 args[
    0] = this.textBox_CityName.Text ;
                 args[
    1] = "China" ;
                
    object result = WebServiceHelper.InvokeWebService(url ,"GetWeather" ,args) ;
                
    this.label_Result.Text = result.ToString() ;


         上述的例子中,调用web服务使用了两个参数,第一个是城市的名字,第二个是国家的名字,Web服务返回的是XML文档,可以从其中解析出温度、风力等天气情况。
        
         最后说一下,C#虽然仍属于静态语言之列,但是其动态能力也是很强大的,不信,你可以看看Spring.net的AOP实现,这种“无侵入”的AOP实现比通常的.NET声明式AOP实现(一般是通过AOP Attribute)要漂亮的多。

    posted on 2010-10-27 22:51  powerwang  阅读(812)  评论(0)    收藏  举报

    导航