通常我们在程序中需要调用WebService时,都是通过“添加Web引用”,让VS.NET环境来为我们生成服务代理,然后调用对应的Web服务。这样是使工作简单了,但是却和提供Web服务的URL、方法名、参数绑定在一起了,这是VS.NET自动为我们生成Web服务代理的限制。如果哪一天发布Web服务的URL改变了,则我们需要重新让VS.NET生成代理,并重新编译。在某些情况下,这可能是不能忍受的,我们需要C#中动态调用WebService的能力。比如我们可以把Web服务的URL保存在配置文件中,这样,当服务URL改变时,只需要修改配置文件就可以了。
说了这么多,实际上我们要实现这样的功能:
- public static object InvokeWebService(string url,
- string methodname, object[] args)
其中,url是Web服务的地址,methodname是要调用服务方法名,args是要调用Web服务所需的参数,返回值就是web服务返回的结果了。
要实现这样的功能,你需要这几个方面的技能:反射、CodeDom、编程使用C#编译器、WebService。在了解这些知识后,就可以容易的实现web服务的动态调用了:
- usingSystem.CodeDom.Compiler;
- usingSystem;
- usingSystem.Net;
- usingSystem.CodeDom;
- usingMicrosoft.CSharp;
- usingSystem.IO;
- usingSystem.Web.Services.Description;
- usingSystem.Collections.Generic;
- usingSystem.Reflection;
- namespacecjl.WebServices
- {
- publicclassDynamicWebServices
- {
- staticSortedList〈string,Type〉_typeList=
- newSortedList〈string,Type〉();
- #regionInvokeWebService
- staticstringGetCacheKey(stringurl,
- stringclassName)
- {
- returnurl.ToLower()+className;
- }
- staticTypeGetTypeFromCache(stringurl,
- stringclassName)
- {
- stringkey=GetCacheKey(url,className);
- foreach(KeyValuePair〈string,Type〉
- pairin_typeList)
- {
- if(key==pair.Key)
- {
- returnpair.Value;
- }
- }
- returnnull;
- }
- staticTypeGetTypeFromWebService
- (stringurl,stringclassName)
- {
- string@namespace="EnterpriseServerBase.
- WebService.DynamicWebCalling";
- if((className==null)||(className==""))
- {
- className=GetWsClassName(url);
- }
- //获取WSDL
- WebClientwc=newWebClient();
- Streamstream=wc.OpenRead(url+"?WSDL");
- ServiceDescriptionsd=ServiceDescription.
- Read(stream);
- ServiceDescriptionImportersdi=
- newServiceDescriptionImporter();
- sdi.AddServiceDescription(sd,"","");
- CodeNamespacecn=newCodeNamespace
- (@namespace);
- //生成客户端代理类代码
- CodeCompileUnitccu=newCodeCompileUnit();
- ccu.Namespaces.Add(cn);
- sdi.Import(cn,ccu);
- CSharpCodeProvidercsc=newCSharpCodeProvider();
- ICodeCompilericc=csc.CreateCompiler();
- //设定编译参数
- CompilerParameterscplist=newCompilerParameters();
- 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");
- //编译代理类
- CompilerResultscr=
- icc.CompileAssemblyFromDom(cplist,ccu);
- if(true==cr.Errors.HasErrors)
- {
- System.Text.StringBuildersb=
- newSystem.Text.StringBuilder();
- foreach(System.CodeDom.Compiler.
- CompilerErrorceincr.Errors)
- {
- sb.Append(ce.ToString());
- sb.Append(System.Environment.NewLine);
- }
- thrownewException(sb.ToString());
- }
- //生成代理实例,并调用方法
- System.Reflection.Assemblyassembly=
- cr.CompiledAssembly;
- Typet=assembly.GetType(@namespace+".
- "+className,true,true);
- returnt;
- }
- //动态调用web服务
- publicstaticobjectInvokeWebService
- (stringurl,stringmethodName,object[]args)
- {
- returnInvokeWebService(url,null,
- methodName,args);
- }
- publicstaticobjectInvokeWebService(stringurl,
- stringclassName,stringmethodName,object[]args)
- {
- try
- {
- Typet=GetTypeFromCache(url,className);
- if(t==null)
- {
- t=GetTypeFromWebService(url,className);
- //添加到缓冲中
- stringkey=GetCacheKey(url,className);
- _typeList.Add(key,t);
- }
- objectobj=Activator.CreateInstance(t);
- MethodInfomi=t.GetMethod(methodName);
- returnmi.Invoke(obj,args);
- }
- catch(Exceptionex)
- {
- thrownewException(ex.InnerException.Message,
- newException(ex.InnerException.StackTrace));
- }
- }
- privatestaticstringGetWsClassName(stringwsUrl)
- {
- string[]parts=wsUrl.Split('/');
- string[]pps=parts[parts.Length-1].Split('.');
- returnpps[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)要漂亮的多。
【编辑推荐】