Winform 動態調用WebService

一般情況下winform調用webservice時步驟
1添加服務引用---高級----添加web引用------填寫url--添加web引用即可完成對webservice的引用
讓VS.NET環境來爲我們生成服務代理,然後調用對應的Web服務。
如果需要動態調用WebService,要實現這樣的功能:
public static object InvokeWebService(string url,  string methodname,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;
using System.Reflection;


namespace VionDataHandler
{
    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 = "EnterpriseServerBase.WebService.DynamicWebCalling";
            if (classname == null || classname == "")
            {
                classname = WebServiceHelper.GetClassName(url);
            }


            //獲取服務描述語言(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();//【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】




            ////生成代理實例,並調用方法   
            //System.Reflection.Assembly assembly1 = cr.CompiledAssembly;
            ////ExcMsg.WriteLog("----------獲取所有的類與方法信息--------" + "。" + "(正常執行!)");
            //Type[] allTypes = assembly.GetTypes();
            //foreach (Type type in allTypes)
            //{
            //    //ExcMsg.WriteLog("--類:" + type.FullName);
            //    MethodInfo[] methods = type.GetMethods();
            //    foreach (MethodInfo info in methods)
            //    {
            //        //ExcMsg.WriteLog(string.Format("----方法:{0}", info.Name));
            //    }
            //}






            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[] allTypes = assembly.GetTypes();
            //foreach (Type type in allTypes)
            //{
            //    //ExcMsg.WriteLog("--類:" + type.FullName);
            //    MethodInfo[] methods = type.GetMethods();
            //    foreach (MethodInfo info in methods)
            //    {
            //        //ExcMsg.WriteLog(string.Format("----方法:{0}", info.Name));
            //    }
            //}


            Type t = assembly.GetType(@namespace + "." + classname, true, true);
            object obj = Activator.CreateInstance(t);//【10】
            MethodInfo mi = t.GetMethod(methodname);//【11】
            //getRegionCountry
            return mi.Invoke(obj, args);
        }


        private static string GetClassName(string url)
        {
            string[] parts = url.Split('/');
            string[] pps = parts[parts.Length - 1].Split('.');
            return pps[0];


        }


    }


}


     上述的例子中,調用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方法來獲取。
發佈了42 篇原創文章 · 獲贊 18 · 訪問量 23萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章