標籤:
一般情況下winform調用webservice時步驟1加入服務參考---進階----添加web引用------填寫url--添加web引用即可完成對webservice的引用讓VS.NET環境來為我們產生服務代理,然後調用對應的Web服務。如果需要動態調用WebService,要實現這樣的功能:publicstaticobjectInvokeWebService(stringurl, stringmethodname,object[] args)其中,url是Web服務的地址,methodname是要調用服務方法名,args是要調用Web服務所需的參數,傳回值就是web服務返回的結果了。 要實現這樣的功能,你需要這幾個方面的技能:反射、CodeDom、編程使用C#編譯器、WebService。在瞭解這些知識後,就可以容易的實現web服務的動態調用了:
using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.IO;using System.Net;using System.CodeDom;using System.CodeDom.Compiler;using System.Web.Services.Description;using Microsoft.CSharp;namespace NetbankTMP{ public static class WebServiceHelper { /// <summary> /// 動態調用WebService /// </summary> /// <param name="url">WebService地址</param> /// <param name="methodname">方法名(模組名)</param> /// <param name="args">參數列表,無參數為null</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 = "fangqm.Netbank.WebService.webservice"; if (classname == null || classname == "") { classname = WebServiceHelper.GetClassName(url); } //擷取服務描述語言(WSDL) WebClient wc = new WebClient(); Stream stream = wc.OpenRead(url+"?WSDL");//【1】 ServiceDescription sd = ServiceDescription.Read(stream);//【2】 ServiceDescriptionImporter sdi = new ServiceDescriptionImporter();//【3】 sdi.AddServiceDescription(sd, "", ""); CodeNamespace cn = new CodeNamespace(@namespace);//【4】 //產生用戶端代理類代碼 CodeCompileUnit ccu = new CodeCompileUnit();//【5】 ccu.Namespaces.Add(cn); sdi.Import(cn, ccu); CSharpCodeProvider csc = new CSharpCodeProvider();//【6】 ICodeCompiler icc = csc.CreateCompiler();//【7】 //設定編譯器的參數 CompilerParameters cplist = new CompilerParameters();//【8】 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);//【9】 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 bj = Activator.CreateInstance(t);//【10】 System.Reflection.MethodInfo mi = t.GetMethod(methodname);//【11】 return mi.Invoke(obj, args); } private static string GetClassName(string url) { //假如URL為"http://localhost/InvokeService/Service1.asmx" //最終的傳回值為 Service1 string[] parts = url.Split(‘/‘); string[] pps = parts[parts.Length - 1].Split(‘.‘); return pps[0]; } }}
View Code
上面的注釋已經很好的說明了各程式碼片段的功能,下面給個例子看看,這個例子是通過訪問http://www.webservicex.net/globalweather.asmx服務來擷取各大城市的天氣狀況。
stringurl="http://www.webservicex.net/globalweather.asmx"; string[] args=newstring[2] ; args[0]=this.textBox_CityName.Text ; args[1]="China"; objectresult=WebServiceHelper.InvokeWebService(url ,"GetWeather",args) ; this.label_Result.Text=result.ToString() ;
View Code
如果沒有參數,則參數為null 上述的例子中,調用web服務使用了兩個參數,第一個是城市的名字,第二個是國家的名字,Web服務返回的是XML文檔,可以從其中解析出溫度、風力等天氣情況。 關於這段代碼的注釋 【2】 ServiceDescription類提供一種方法,以建立和格式化用於描述 XML Web services 的有效 Web 服務描述語言 (WSDL) 文檔檔案,該檔案是完整的,具有適當的命名空間、元素和特性。無法繼承此類。ServiceDescription.Read 方法 (Stream) 通過直接從 Stream執行個體載入 XML 來初始化ServiceDescription類的執行個體。 【3】 ServiceDescriptionImporter 類 公開一種為 XML Web services 產生用戶端代理類的方法。ServiceDescriptionImporter.AddServiceDescription 方法將指定的ServiceDescription添加到要匯入的ServiceDescriptions值的集合中。 【4】 CodeNamespace表示命名空間聲明。 【5】 CodeCompileUnit會提供一個CodeDOM程式圓形的容器,CodeCompileUnit含有一個集合,可以儲存含有CodeDOM原始程式碼原形,專案參考的組件集合以及專案組件屬性集合的CodeNamespace物件。 【6】 CSharpCodeProvider類提供存取C#程式碼產生器和程式碼編譯器的執行個體。 【7】 取得C#程式碼編譯器的執行個體 【8】 建立編譯器的參數執行個體 【9】 CompilerResults表示從編譯器返回的編譯結果。使用指定的編譯器設定,根據CodeCompileUnit物件之指定陣列所包含的System.CodeDom樹狀結構,編譯一個組件。 【10】 Activator類包含特定的方法,用以在本地或從遠程建立物件類型,或擷取對現有遠程對象的引用。無法繼承此類Activator.CreateInstance 方法 使用與指定參數匹配程度最高的建構函式建立指定類型的執行個體。 【11】 MethodInfo 的執行個體可以通過調用GetMethods或者Type對象或派生自Type的對象的GetMethod方法來擷取,還可以通過調用表示泛型方法定義的 MethodInfo 的MakeGenericMethod方法來擷取。
C#中Winform動態調用Webservice的方法(轉)