通常我们在程序中需要调用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服务的动态调用了:
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 服务来获取各大城市的天气状况。
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)要漂亮的多。