Winform動態呼叫Webservice的方法
阿新 • • 發佈:2019-01-05
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];
}
}
}
上面的註釋已經很好的說明了各程式碼段的功能,下面給個例子看看,這個例子是通過訪問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方法來獲取。