动态调用WebService(C#)

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

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

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

C#代码 复制代码
  1. #region InvokeWebService   
  2.        //动态调用web服务   
  3.        public static object InvokeWebService(string url, string methodname, object[] args)   
  4.         {   
  5.            return WebServiceHelper.InvokeWebService(url ,null ,methodname ,args) ;   
  6.         }   
  7.   
  8.        public static object InvokeWebService(string url,  string classname, string methodname, object[] args)   
  9.         {   
  10.            string @namespace = "EnterpriseServerBase.WebService.DynamicWebCalling" ;   
  11.            if((classname == null) ||(classname == ""))   
  12.             {   
  13.                 classname = WebServiceHelper.GetWsClassName(url) ;   
  14.             }   
  15.   
  16.            try  
  17.             {   
  18.                //获取WSDL   
  19.                 WebClient wc                   = new WebClient();   
  20.                 Stream stream                  = wc.OpenRead(url+"?WSDL");   
  21.                 ServiceDescription sd          = ServiceDescription.Read(stream);   
  22.                 ServiceDescriptionImporter sdi = new ServiceDescriptionImporter();   
  23.                 sdi.AddServiceDescription(sd,"","");   
  24.                 CodeNamespace cn                = new CodeNamespace(@namespace);   
  25.                   
  26.                //生成客户端代理类代码   
  27.                 CodeCompileUnit ccu             = new CodeCompileUnit();   
  28.                 ccu.Namespaces.Add(cn);   
  29.                 sdi.Import(cn ,ccu);    
  30.                 CSharpCodeProvider csc          = new CSharpCodeProvider();   
  31.                 ICodeCompiler icc               = csc.CreateCompiler();   
  32.                   
  33.                //设定编译参数   
  34.                 CompilerParameters cplist       = new CompilerParameters();   
  35.                 cplist.GenerateExecutable       = false;   
  36.                 cplist.GenerateInMemory         = true;   
  37.                 cplist.ReferencedAssemblies.Add("System.dll");   
  38.                 cplist.ReferencedAssemblies.Add("System.XML.dll");   
  39.                 cplist.ReferencedAssemblies.Add("System.Web.Services.dll");   
  40.                 cplist.ReferencedAssemblies.Add("System.Data.dll");   
  41.   
  42.                //编译代理类   
  43.                 CompilerResults cr = icc.CompileAssemblyFromDom(cplist, ccu);   
  44.                if(true == cr.Errors.HasErrors)   
  45.                 {   
  46.                     System.Text.StringBuilder sb = new System.Text.StringBuilder();   
  47.                    foreach(System.CodeDom.Compiler.CompilerError ce in cr.Errors)   
  48.                     {   
  49.                         sb.Append(ce.ToString());   
  50.                         sb.Append(System.Environment.NewLine);   
  51.                     }   
  52.                    throw new Exception(sb.ToString());   
  53.                 }   
  54.   
  55.                //生成代理实例,并调用方法   
  56.                 System.Reflection.Assembly assembly = cr.CompiledAssembly;   
  57.                 Type t = assembly.GetType(@namespace+"."+classname,true,true);   
  58.                object obj = Activator.CreateInstance(t);   
  59.                 System.Reflection.MethodInfo mi = t.GetMethod(methodname);   
  60.   
  61.                return mi.Invoke(obj,args);   
  62.             }   
  63.            catch(Exception ex)   
  64.             {   
  65.                throw new Exception(ex.InnerException.Message,new Exception(ex.InnerException.StackTrace));   
  66.             }   
  67.         }   
  68.   
  69.        private static string GetWsClassName(string wsUrl)   
  70.         {   
  71.            string[] parts = wsUrl.Split('/') ;   
  72.            string[] pps   = parts[parts.Length-1].Split('.') ;   
  73.   
  74.            return pps[0] ;   
  75.         }  
  76.        #endregion  

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

C#代码 复制代码
  1. string url = "http://www.webservicex.net/globalweather.asmx" ;   
  2.        string[] args = new string[2] ;   
  3.         args[0] = this.textBox_CityName.Text ;   
  4.         args[1] = "China" ;   
  5.        object result = WebServiceHelper.InvokeWebService(url ,"GetWeather" ,args) ;   
  6.        this.label_Result.Text = result.ToString() ;  

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

原文地址:https://www.cnblogs.com/bingyun84/p/1926020.html