动态调用WebService(C#)

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


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

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

[csharp] view plain copy
print ?
  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  
  #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 服务来获取各大城市的天气状况。

[csharp] view plain copy
print ?
  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() ;  
     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)要漂亮的多。

c#动态调用WebService

[csharp] view plain copy
print ?
  1. using System;  
  2. using System.Collections.Generic;  
  3. using System.Text;  
  4. using System.Xml;  
  5. using System.Net;  
  6. using System.Web.Services.Description;  
  7. using System.CodeDom;  
  8. using System.CodeDom.Compiler;  
  9. using System.Reflection;  
  10. namespace WindowsServiceWebDefaultHotCity  
  11. {  
  12.     /// <summary<  
  13.     /// WebService代理类  
  14.     /// </summary<  
  15.     public class WebServiceAgent  
  16.     {  
  17.         private object agent;  
  18.         private Type agentType;  
  19.         private const string CODE_NAMESPACE = "Beyondbit.WebServiceAgent.Dynamic";  
  20.         /// <summary<  
  21.         /// 构造函数  
  22.         /// </summary<  
  23.         /// <param name="url"<</param<  
  24.         public WebServiceAgent(string url)  
  25.         {  
  26.             XmlTextReader reader = new XmlTextReader(url + "?wsdl");  
  27.   
  28.             //创建和格式化 WSDL 文档  
  29.             ServiceDescription sd = ServiceDescription.Read(reader);  
  30.   
  31.             //创建客户端代理代理类  
  32.             ServiceDescriptionImporter sdi = new ServiceDescriptionImporter();  
  33.             sdi.AddServiceDescription(sd, nullnull);  
  34.   
  35.             //使用 CodeDom 编译客户端代理类  
  36.             CodeNamespace cn = new CodeNamespace(CODE_NAMESPACE);  
  37.             CodeCompileUnit ccu = new CodeCompileUnit();  
  38.             ccu.Namespaces.Add(cn);  
  39.             sdi.Import(cn, ccu);  
  40.             Microsoft.CSharp.CSharpCodeProvider icc = new Microsoft.CSharp.CSharpCodeProvider();  
  41.             CompilerParameters cp = new CompilerParameters();  
  42.             CompilerResults cr = icc.CompileAssemblyFromDom(cp, ccu);  
  43.             agentType = cr.CompiledAssembly.GetTypes()[0];  
  44.             agent = Activator.CreateInstance(agentType);  
  45.         }  
  46.   
  47.         ///<summary<  
  48.         ///调用指定的方法  
  49.         ///</summary<  
  50.         ///<param name="methodName"<方法名,大小写敏感</param<  
  51.         ///<param name="args"<参数,按照参数顺序赋值</param<  
  52.         ///<returns<Web服务的返回值</returns<  
  53.         public object Invoke(string methodName, params object[] args)  
  54.         {  
  55.             MethodInfo mi = agentType.GetMethod(methodName);  
  56.             return this.Invoke(mi, args);  
  57.         }  
  58.         ///<summary<  
  59.         ///调用指定方法  
  60.         ///</summary<  
  61.         ///<param name="method"<方法信息</param<  
  62.         ///<param name="args"<参数,按照参数顺序赋值</param<  
  63.         ///<returns<Web服务的返回值</returns<  
  64.         public object Invoke(MethodInfo method, params object[] args)  
  65.         {  
  66.             return method.Invoke(agent, args);  
  67.         }  
  68.         public MethodInfo[] Methods  
  69.         {  
  70.             get  
  71.             {  
  72.                 return agentType.GetMethods();  
  73.             }  
  74.         }  
  75.     }  
  76. }  
using System;
using System.Collections.Generic;
using System.Text;
using System.Xml;
using System.Net;
using System.Web.Services.Description;
using System.CodeDom;
using System.CodeDom.Compiler;
using System.Reflection;
namespace WindowsServiceWebDefaultHotCity
{
    /// <summary<
    /// WebService代理类
    /// </summary<
    public class WebServiceAgent
    {
        private object agent;
        private Type agentType;
        private const string CODE_NAMESPACE = "Beyondbit.WebServiceAgent.Dynamic";
        /// <summary<
        /// 构造函数
        /// </summary<
        /// <param name="url"<</param<
        public WebServiceAgent(string url)
        {
            XmlTextReader reader = new XmlTextReader(url + "?wsdl");

            //创建和格式化 WSDL 文档
            ServiceDescription sd = ServiceDescription.Read(reader);

            //创建客户端代理代理类
            ServiceDescriptionImporter sdi = new ServiceDescriptionImporter();
            sdi.AddServiceDescription(sd, null, null);

            //使用 CodeDom 编译客户端代理类
            CodeNamespace cn = new CodeNamespace(CODE_NAMESPACE);
            CodeCompileUnit ccu = new CodeCompileUnit();
            ccu.Namespaces.Add(cn);
            sdi.Import(cn, ccu);
            Microsoft.CSharp.CSharpCodeProvider icc = new Microsoft.CSharp.CSharpCodeProvider();
            CompilerParameters cp = new CompilerParameters();
            CompilerResults cr = icc.CompileAssemblyFromDom(cp, ccu);
            agentType = cr.CompiledAssembly.GetTypes()[0];
            agent = Activator.CreateInstance(agentType);
        }

        ///<summary<
        ///调用指定的方法
        ///</summary<
        ///<param name="methodName"<方法名,大小写敏感</param<
        ///<param name="args"<参数,按照参数顺序赋值</param<
        ///<returns<Web服务的返回值</returns<
        public object Invoke(string methodName, params object[] args)
        {
            MethodInfo mi = agentType.GetMethod(methodName);
            return this.Invoke(mi, args);
        }
        ///<summary<
        ///调用指定方法
        ///</summary<
        ///<param name="method"<方法信息</param<
        ///<param name="args"<参数,按照参数顺序赋值</param<
        ///<returns<Web服务的返回值</returns<
        public object Invoke(MethodInfo method, params object[] args)
        {
            return method.Invoke(agent, args);
        }
        public MethodInfo[] Methods
        {
            get
            {
                return agentType.GetMethods();
            }
        }
    }
}


 

[csharp] view plain copy
print ?
  1. using System;  
  2. using System.Collections.Generic;  
  3. using System.ComponentModel;  
  4. using System.Data;  
  5. using System.Drawing;  
  6.   
  7. using System.Text;  
  8. using System.Windows.Forms;  
  9.   
  10. namespace WindowsApplication1  
  11. {  
  12.     public partial class Form1 : Form  
  13.     {  
  14.         private string _url = "http://www.baidu.com";  
  15.         public Form1()  
  16.         {  
  17.             InitializeComponent();  
  18.             init_Data();  
  19.         }  
  20.   
  21.         public void init_Data()  
  22.         {  
  23.             WindowsServiceWebDefaultHotCity.WebServiceAgent agent = new WindowsServiceWebDefaultHotCity.WebServiceAgent(_url);  
  24.             object[] args = new object[6];  
  25.             args[0] = "PEK";  
  26.             args[1] = "CAN";  
  27.             args[2] = "";  
  28.             args[3] = "2008-08-02";  
  29.             args[4] = "00:00";  
  30.             args[5] = "own_9588";  
  31.             string text=agent.Invoke("GetAllFlight", args).ToString();  
  32.             textBox1.Text = text;  
  33.         }  
  34.     }  
  35. }  
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;

using System.Text;
using System.Windows.Forms;

namespace WindowsApplication1
{
    public partial class Form1 : Form
    {
        private string _url = "http://www.baidu.com";
        public Form1()
        {
            InitializeComponent();
            init_Data();
        }

        public void init_Data()
        {
            WindowsServiceWebDefaultHotCity.WebServiceAgent agent = new WindowsServiceWebDefaultHotCity.WebServiceAgent(_url);
            object[] args = new object[6];
            args[0] = "PEK";
            args[1] = "CAN";
            args[2] = "";
            args[3] = "2008-08-02";
            args[4] = "00:00";
            args[5] = "own_9588";
            string text=agent.Invoke("GetAllFlight", args).ToString();
            textBox1.Text = text;
        }
    }
}
 

   该方法可以使程序不通过web引用的方式去调用webservices方法,直接在代码里调用该方法就能达到动态调用webservices的目的。使用前先引用System.Web.Services动态链接库,是.net自带的dll。

方法如下:

[csharp] view plain copy
print ?
  1. using System;  
  2. using System.Collections.Generic;  
  3. using System.Text;  
  4. using System.Net;  
  5. using System.IO;  
  6. using System.Web.Services.Description;  
  7. using System.CodeDom;  
  8. using Microsoft.CSharp;  
  9. using System.CodeDom.Compiler;  
  10.   
  11. namespace TestSkin  
  12. {  
  13.     class Webservices  
  14.     {  
  15.         /// <summary<  
  16.         /// 实例化WebServices  
  17.         /// </summary<  
  18.         /// <param name="url"<WebServices地址</param<  
  19.         /// <param name="methodname"<调用的方法</param<  
  20.         /// <param name="args"<把webservices里需要的参数按顺序放到这个object[]里</param<  
  21.         public static object InvokeWebService(string url, string methodname, object[] args)  
  22.         {  
  23.   
  24.             //这里的namespace是需引用的webservices的命名空间,在这里是写死的,大家可以加一个参数从外面传进来。  
  25.             string @namespace = "client";  
  26.             try  
  27.             {  
  28.                 //获取WSDL  
  29.                 WebClient wc = new WebClient();  
  30.                 Stream stream = wc.OpenRead(url + "?WSDL");  
  31.                 ServiceDescription sd = ServiceDescription.Read(stream);  
  32.                 string classname = sd.Services[0].Name;  
  33.                 ServiceDescriptionImporter sdi = new ServiceDescriptionImporter();  
  34.                 sdi.AddServiceDescription(sd, """");  
  35.                 CodeNamespace cn = new CodeNamespace(@namespace);  
  36.   
  37.                 //生成客户端代理类代码  
  38.                 CodeCompileUnit ccu = new CodeCompileUnit();  
  39.                 ccu.Namespaces.Add(cn);  
  40.                 sdi.Import(cn, ccu);  
  41.                 CSharpCodeProvider csc = new CSharpCodeProvider();  
  42.                 ICodeCompiler icc = csc.CreateCompiler();  
  43.   
  44.                 //设定编译参数  
  45.                 CompilerParameters cplist = new CompilerParameters();  
  46.                 cplist.GenerateExecutable = false;  
  47.                 cplist.GenerateInMemory = true;  
  48.                 cplist.ReferencedAssemblies.Add("System.dll");  
  49.                 cplist.ReferencedAssemblies.Add("System.XML.dll");  
  50.                 cplist.ReferencedAssemblies.Add("System.Web.Services.dll");  
  51.                 cplist.ReferencedAssemblies.Add("System.Data.dll");  
  52.   
  53.                 //编译代理类  
  54.                 CompilerResults cr = icc.CompileAssemblyFromDom(cplist, ccu);  
  55.                 if (true == cr.Errors.HasErrors)  
  56.                 {  
  57.                     System.Text.StringBuilder sb = new System.Text.StringBuilder();  
  58.                     foreach (System.CodeDom.Compiler.CompilerError ce in cr.Errors)  
  59.                     {  
  60.                         sb.Append(ce.ToString());  
  61.                         sb.Append(System.Environment.NewLine);  
  62.                     }  
  63.                     throw new Exception(sb.ToString());  
  64.                 }  
  65.   
  66.                 //生成代理实例,并调用方法  
  67.                 System.Reflection.Assembly assembly = cr.CompiledAssembly;  
  68.                 Type t = assembly.GetType(@namespace + "." + classname, truetrue);  
  69.                 object obj = Activator.CreateInstance(t);  
  70.                 System.Reflection.MethodInfo mi = t.GetMethod(methodname);  
  71.   
  72.                 return mi.Invoke(obj, args);  
  73.             }  
  74.             catch  
  75.             {  
  76.                 return null;  
  77.             }  
  78.         }  
  79.     }  
  80. }  
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值