通常我們在程序中需要調用WebService時,都是通過“添加Web引用”,讓
VS.NET環境來爲我們生成服務代理,然後調用對應的Web服務。這樣是使工作簡單了,但是卻和提供Web服務的URL、方法名、參數綁定在一起了,這
是VS.NET自動爲我們生成Web服務代理的限制。如果哪一天發佈Web服務的URL改變了,則我們需要重新讓VS.NET生成代理,並重新編譯。在某
些情況下,這可能是不能忍受的,我們需要動態調用WebService的能力。比如我們可以把Web服務的URL保存在配置文件中,這樣,當服務URL改
變時,只需要修改配置文件就可以了。
說了這麼多,實際上我們要實現這樣的功能:
其中,url是Web服務的地址,methodname是要調用服務方法名,args是要調用Web服務所需的參數,返回值就是web服務返回的結果了。
要實現這樣的功能,你需要這幾個方面的技能:反射、CodeDom、編程使用C#編譯器、WebService。在瞭解這些知識後,就可以容易的實現web服務的動態調用了:
// 動態調用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
服務來獲取各大城市的天氣狀況。
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)要漂亮的多。