动态调用webservice

using System;
using System.Collections.Generic;
using System.Text;

using System.IO;
using System.Configuration;
// 命名空间包含提供用于处理配置数据的编程模型的类型。 
using System.CodeDom;
/*代码文档对象模型”, 使用该模型建立的代码文档可以被.NET Framework编译成应用程序集。也就是说,
 *你可以使用该模型“发明”一个自己的.net语言,用你的语言编写程序,再翻译成codeDom,
 * 最后编译成可以执行的.net应用程序。
 *归根结底在System.CodeDom这个命名空间里你new来new去只能是得到一些Object而已,
 * 而Object是什么呢,就是内存里的一点点数据
 */
using System.CodeDom.Compiler;
/*System.CodeDom.Compiler这个命名空间里的东西才能把它表现出来。前一个命名空间在于构造,
 * 后一个命名空间在于表现。构造就是搭个架子,把里面的各个部分聚合聚合,连接连接,
 * 这个一点点奥秘都没有,所有也不去深究了
 */
using System.Net;//利用System.Net类访问因特网
using System.Web.Services;
using System.Web.Services.Description;
/* 命名空间由使得您可以通过使用 Web 服务描述语言 (WSDL) 来公开描述 XML Web services 的类组成。
 * 此命名空间中的每个类都与 WSDL 规范中的某个特定元素相对应,
 * 并且类的层次结构与有效的 WSDL 文档的 XML 结构相对应。
 * 有关 WSDL 的更多信息,请参见位于 W3C 网站 (http://www.w3.org/TR/wsdl/) 的规范。
 */
using Microsoft.CSharp;
//命名空间包含支持使用 C# 语言编译和生成代码的类。

namespace InvokeWebService
{
    public static class WebServiceHelper
    {
        /// <summary>
        /// 动态调用WebService
        /// </summary>
        /// <param name="url">WebService地址</param>
        /// <param name="methodname">方法名(模块名)</param>
        /// <param name="args">参数列表</param>
        /// <returns>object</returns>
        public static object InvokeWebService(string url, string methodname, object[] args)
        {
            return InvokeWebService(url, null, methodname, args);
        }
        /// <summary>
        /// 动态调用WebService
        /// </summary>
        /// <param name="url">WebService地址</param>
        /// <param name="classname">类名</param>
        /// <param name="methodname">方法名(模块名)</param>
        /// <param name="args">参数列表</param>
        /// <returns>object</returns>
        public static object InvokeWebService(string url, string classname, string methodname, object[] args)
        {
            string @namespace = "ServiceBase.WebService.DynamicWebLoad";
            if (classname == null || classname == "")
            {
                classname = WebServiceHelper.GetClassName(url);
            }
            //(一):获取服务描述语言(WSDL)

            //提供向 URI 标识的资源发送数据和从 URI 标识的资源接收数据的公共方法
            WebClient wc = new WebClient();
            Stream stream = wc.OpenRead(url + "?WSDL");
            /*下表描述从资源下载数据的 WebClient 方法.相对应的是上载到资源的 WebClient 方法。
             * OpenRea 资源以 Stream 的形式返回数据,对应与OpenWrite()
             * OpenReadAsync 在不阻止调用线程的情况下,从资源返回数据
             * DownloadData从资源下载数据并返回 Byte 数组。
             * DownloadDataAsync在不阻止调用线程的情况下,从资源下载数据并返回 Byte 数组。
             * DownloadFile从资源将数据下载到本地文件。
             * DownloadFileAsync在不阻止调用线程的情况下,将数据从资源下载到本地文件。
             * DownloadString从资源下载 String 并返回 String。
             * DownloadStringAsync在不阻止调用线程的情况下,从资源下载 String
             */

            //ServiceDescription 类与 WSDL 文件的根元素 definitions 相对应。
            ServiceDescription sd = ServiceDescription.Read(stream);

            //使用 ServiceDescriptionImporter 类可以方便地将 WSDL 说明中包含的信息导入到 
            //System.CodeDom.CodeCompileUnit 对象
            ServiceDescriptionImporter sdi = new ServiceDescriptionImporter();
            sdi.AddServiceDescription(sd, "", "");
            CodeNamespace cn = new CodeNamespace(@namespace);

            //(二):生成客户端代理类代码

            /*CodeCompileUnit 包含以下几个集合:可以存储包含
             * CodeDOM 源代码图形的 CodeNamespace 对象的集合、项目引用的程序集的集合,
             * 以及项目程序集的属性集合。
             */
            CodeCompileUnit ccu = new CodeCompileUnit();

            // Add the new namespace to the compile unit.
            ccu.Namespaces.Add(cn);

            //导入指定的 ServiceDescriptions 值,并将按照 Style 属性的指定来生成代码。 
            // Add the new namespace import for the System namespace
            sdi.Import(cn, ccu);

            //提供对 C# 代码生成器和代码编译器的实例的访问。
            CSharpCodeProvider csc = new CSharpCodeProvider();

            /*ICodeCompiler 接口提供用于在运行时使用指定的参数调用编译的功能,
              *以及在编译之后访问编译相关信息的功能,这些信息包括结果代码和编译器返回的任何错误或警告。
              * 每种编译方法都接受指示编译器设置的 CompilerParameters 对象,并返回指示编译结果的 CompilerResults 对象。
              */
            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 StringBuilder();
                foreach (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);

            //发现方法的属性 (Attribute) 并提供对方法元数据的访问。 
            System.Reflection.MethodInfo mi = t.GetMethod(methodname);
            return mi.Invoke(obj, args);

        }

        private static string GetClassName(string url)
        {
            string[] parts = url.Split('/');
            string[] pps = parts[parts.Length - 1].Split('.');
            return pps[0];
        }

    }
}

你可能感兴趣的:(webservice,动态)