C#中Winform動態呼叫Webservice的方法
阿新 • • 發佈:2019-01-22
一般情況下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服務的動態呼叫了:
上面的註釋已經很好的說明了各程式碼段的功能,下面給個例子看看,這個例子是通過訪問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() ; 如果沒有引數,則引數為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方法來獲取。
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]; } } }
上面的註釋已經很好的說明了各程式碼段的功能,下面給個例子看看,這個例子是通過訪問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() ; 如果沒有引數,則引數為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方法來獲取。