我们都知道,调用WS可以在工程中添加对WS的WEB引用。
但是,如果我们不想通过添加引用的方式,而是在代码中动态引用该怎么办呢?
首先,我们该想到WS的实现也是一个类的形式。
其次,WS在传输过程中是通过WSDL来进行描述的(使用SOAP协议)。
因此,我们需要获取WS的WSDL描述,并通过该描述来动态生成程序集。
最后:通过反射来获取新生成的程序集,并调用其方法!
上述步骤需要引用如下四个名称空间:
using System.Web.Services.Description; //WS的描述
//以下几个用于根据描述动态生成代码并动态编译获取程序集
using System.CodeDom;
using Microsoft.CSharp;
using System.CodeDom.Compiler;
上述几个名称空间中包括如下几个重要的类:
using System.Web.Services.Description下:
ServiceDescription //WS描述
ServiceDescriptionImporter //通过描述生成客户端代理类,特别注意其中的Style
以下是MSDN对其的描述:
XML Web services 的接口通常由 Web 服务描述语言 (WSDL) 文件来说明。例如,若要获取有关使用 http://localhost/service.asmx
处公开的 ASP.NET 的 Web 服务的 WSDL 说明,只需导航到 http://localhost/service.asmx?WSDL
。使用 ServiceDescriptionImporter 类可以方便地将 WSDL 说明中包含的信息导入到
using System.CodeDom下:
CodedomUnit //它用于设定动态代码的名称空间,类名等,可以通过ServiceDescriptionImporter.Import()方法将WS的描述代码写入该类,以作动态编译用
using System.CodeDom.Compiler下:
CodedomProvider //用于创建和检索代码生成器和代码编译器的实例,我们主要用到其实现子类CShareCodeProvider
可以直接用CShareCodeProvider provider=new CShareCodeProvider()来生成,或者用CodedomProvider.CreateProvider("CSharp")来生成
ICodeCompiler //用于编译基于 System.CodeDom 的源代码表示形式。
它通过CodedomProvider的CreateCompiler()方法来
CompilerResults //表示从编译器返回的编译结果。 它由ICodeCompiler根据指定的编译器设置从指定的 CodeCompileUnit 所包含的 System.CodeDom 树中编译程序集并返回。CompiledAssembly 属性指示编译的程序集。
例子:
2 public static object InvokeWebService(string url, string methodname, object[] args)
3 {
4 return WebServiceHelper.InvokeWebService(url, null, methodname, args);
5 }
6
7 public static object InvokeWebService(string url, string classname, string methodname, object[] args)
8 {
9 string @namespace = "EnterpriseServerBase.WebService.DynamicWebCalling";
10 if ((classname == null) || (classname == ""))
11 {
12 classname = WebServiceHelper.GetWsClassName(url);
13 }
14
15 try
16 {
17 //获取WSDL
18 WebClient wc = new WebClient();
19 Stream stream = wc.OpenRead(url + "?WSDL");
20
21 //创建和格式化WSDL
22 ServiceDescription sd = ServiceDescription.Read(stream);
23 ServiceDescriptionImporter sdi = new ServiceDescriptionImporter();
24 sdi.AddServiceDescription(sd, "", "");
25 CodeNamespace cn = new CodeNamespace(@namespace);
26
27 //生成客户端代理类代码
28 CodeCompileUnit ccu = new CodeCompileUnit();
29 ccu.Namespaces.Add(cn);
30 sdi.Import(cn, ccu);
31 CSharpCodeProvider csc = new CSharpCodeProvider();
32 ICodeCompiler icc = csc.CreateCompiler();
33
34 //设定编译参数
35 CompilerParameters cplist = new CompilerParameters();
36 cplist.GenerateExecutable = false;
37 cplist.GenerateInMemory = true;
38 cplist.ReferencedAssemblies.Add("System.dll");
39 cplist.ReferencedAssemblies.Add("System.XML.dll");
40 cplist.ReferencedAssemblies.Add("System.Web.Services.dll");
41 cplist.ReferencedAssemblies.Add("System.Data.dll");
42
43 //编译代理类
44 CompilerResults cr = icc.CompileAssemblyFromDom(cplist, ccu);
45 if (true == cr.Errors.HasErrors)
46 {
47 System.Text.StringBuilder sb = new System.Text.StringBuilder();
48 foreach (System.CodeDom.Compiler.CompilerError ce in cr.Errors)
49 {
50 sb.Append(ce.ToString());
51 sb.Append(System.Environment.NewLine);
52 }
53 throw new Exception(sb.ToString());
54 }
55
56 //生成代理实例,并调用方法
57 System.Reflection.Assembly assembly = cr.CompiledAssembly;
58 Type t = assembly.GetType(@namespace + "." + classname, true, true);
59 object obj = Activator.CreateInstance(t);
60 System.Reflection.MethodInfo mi = t.GetMethod(methodname);
61
62 return mi.Invoke(obj, args);
63 }
64 catch (Exception ex)
65 {
66 throw new Exception(ex.InnerException.Message, new Exception(ex.InnerException.StackTrace));
67 }
68 }
69
70 private static string GetWsClassName(string wsUrl)
71 {
72 string[] parts = wsUrl.Split('/');
73 string[] pps = parts[parts.Length - 1].Split('.');
74
75 return pps[0];
76 }