代码改变世界

动态调用webservice

2009-07-28 10:57  lome  阅读(4025)  评论(21编辑  收藏  举报
  通常我们在程序中需要调用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)要漂亮的多。