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];
}
}
}