首先,我們必須了解什麼是webservice.就概念上來說,可能比較復雜,不過我們可以有個宏觀的了解:webservice就是個對外的接口,裡面有 函數可供外部客戶調用(注意:裡面同樣有客戶不可調用的函數).假若我們是服務端,我們寫好了個webservice,然後把它給了客戶(同時我們給了他 們調用規則),客戶就可以在從服務端獲取信息時處於一個相對透明的狀態.即是客戶不了解(也不需要)其過程,他們只獲取數據.
webservice傳遞的數據只能是序列化的數據,典型的就是xml數據,這裡我們也只討論XML數據的傳輸.
有了一些對XML webservice的初步了解後,我們將切入正題,即是用一個具體的webservice事例的形式來講解具體的webservice用法,用具體的事例來講解一個概念我想怎麼也要比單純的說理能讓人容易理解吧.
這裡,我們將以一個簡單的分布式課件搜索系統為例來講解.使用VS2003為編譯環境,C#為語言,SqlServcer2000為數據庫.(這個例子來 源於一位網上朋友的文章的啟發,覺得很能代表webservice的特點,就按那個想法做了這麼個系統來示例了)
首先,明確我們要做什麼.我們需要一個對客戶的接口,也就是個站點,我們把它稱做ServiceGatherSite,它是何種形式都無所謂,甚至它本身 並不需要數據庫,它只是提供給用戶一個查詢的接口,真正的服務,普通用戶是不接觸到的.然後,這裡我們還需要若干個提供服務的站點,我們可以稱它們為資源 站,這裡為簡單起見,假設有兩個資源站,分別叫WebSiteA,WebSiteB,它們可以是不對外公布的,只是為了豐富查詢數據而存在.最後,是我們 最需要關注的東西---資源站提供給ServiceGatherSite的服務.兩個資源站,就有兩個服務,我們稱為SiteAService和 SiteBService.兩個服務間沒有任何關系,內部提供的方法也完全沒關聯,只是需要把方法如何使用告訴ServiceGatherSite,意思 是,服務只提供查詢接口,返回的數據如何處理,服務本身並不管,全由使用服務的站點分配.
寫了這麼多,算是簡要的介紹了下有關XML webservice的概念和我們這個例子的結構,下篇文章,我們將開始真正進入代碼的設計階段.
上篇文章介紹了些webservice的基本特性和我們例子的結構,這篇文章我們將開始具體的代碼編寫工作.
這個專題主要講述的是webservice,因此這裡我們的代碼以Webservice相關為主,而其他工程,例如:ServiceGatherSite,WebSiteA等,只將簡略介紹.
在VS2003中,開發一個webservice並不是件困難的事,首先,我們新建一個webservice項目(文件->新建->項目->C#->Web服務應用程序)
建完這個工程,我們將看到一個叫Service1.asmx的文件,這就是webservice的標准文件,它也有UI的概念,不過我們一般不關注,因 此,我們查看其cs代碼文件.如果你什麼都還沒做的話,將看見一個被注釋掉的helloworld的WebMethod,把注釋去掉,在運行,你就可以得 到最簡單的webservice運行實例了.點擊"helloworld"將執行其方法.顯然,這個函數對我們的意義只在於宏觀的了解了下web服務的寫 法.
下面,我們將開始具體介紹webservice的寫法.在代碼文件裡,如果我們寫了一個函數後,希望此函數成為外部可調用的接口函數,我們必須在函數上面 添上一行代碼[WebMethod(Description="函數的描述信息")],如果你的函數沒有這個申明,它將不能被用戶引用.如:
以下是引用片段:
[WebMethod(Description="最簡單的方法")] public string HelloWorld() { return "Hello World"; }
這個函數就是外部可調用的接口函數,對用戶來說相當於一個API.如果某用戶在引用了這個服務後,他調用HelloWorld()方法,他就將獲得"HelloWorld"這個返回值.
看到這裡,我們是不是發現,其實webservice並不是那麼的神秘,它也不過只是個接口,對我們而言,側重點依然是接口函數的編寫.下面,我將給出我們的例子所需要的接口函數.
[WebMethod(Description="查詢以獲取需要的課件信息")] public XmlDataDocument GetSiteAData(string AssignName) { XmlDataDocument xd=new XmlDataDocument(); // DataSet ds=new DataSet(); CStoreProc cp=new CStoreProc("SearchAssign"); cp.AddParIn("@keyWords",SqlDbType.VarChar,30,AssignName); cp.AddParOut("@res",SqlDbType.Int); if(cp.SelectProc()) //如果執行成功,存儲過程 { cp.myData.EnforceConstraints=false; //不進行格式嚴格檢查 if((int)cp.GetReturnValue("@res")==-1) { string xml=""; xd.LoadXml(xml); return xd; } xd=new XmlDataDocument(cp.myData); XmlNode root1=xd.DocumentElement; XmlNodeList roots=root1.SelectNodes("list"); foreach(XmlNode roota in roots) //為所有元素加上站點名稱標記 { XMLElement Link=xd.CreateElement("SiteName"); Link.InnerText=ConfigurationSettings.APPSettings["SiteName"].ToString(); roota.AppendChild(Link); } return xd; } else return null; }
這是獲取資源站點信息的一個接口函數.裡面大部分的代碼,我想對於有一定ASP.Net基礎的朋友來說,都應該是一看就明白,這裡只說明下CStoreProc,這是我封裝的一個存儲過程類,主要功能是執行各種類型的存儲過程.
細心的朋友可能會發現這個函數的返回類型似乎比較特殊,是個xml的文檔.我們在前面已經說過,webservice只能傳輸序列化數據,xml顯然滿足 條件,但比如hash表之類的非序列化數據,是不能傳輸的,xml使用最為廣泛,而且考慮到跨平台應用,所以這裡我們只以XML數據的傳輸來示例.
接上篇文章,我們先簡單解釋下GetSiteAData(string AssignName)函數.
函數功能很簡單,只是要返回查詢結果,其數據格式是XmlDataDocument.當查詢失敗時(無匹配查詢結果),我們構造一個xml,返回一個空記 錄.否則,我們把查詢後的dataset生成一個XmlDataDocument,接下來,由於該項目的需要,我加入了一個循環,添加dataset裡所 沒有的節點,站點名稱.在這之後,算是完成了一個符合我們期望格式的XML數據文檔,我們把它返回.
好了,webservice的方法函數介紹完了(這裡還有個web服務方法,稍後介紹),接下來我們的任務是怎麼調用它了.首先把webservice的 項目編譯完成,假定我們這個服務是針對資源站點A的,我們不妨稱其為ServiceA.先單獨運行asmx文件,執行GetSiteAData (string AssignName)方法,將提示你輸入參數,你輸入要搜索的內容,點確認,將返回給你一個XML數據,並在IE上顯示出來,這就是你搜索到的內容拉.
這裡對ServiceA的工作再做點介紹,在我們這個項目裡,它是資源站點A提供的服務,意思是,它查詢的數據將全來源於站點A,而站點A資源添加在本項目也有專門的工程實現.
好了,回到正題.這裡我介紹vs調用webservice的方法,其實其他平台的調用方法也是大同小異.首先我們介紹web引用方式,這種方式我強烈建議 調試時使用,非常方便.右擊引用,點添加web引用,輸入你的webservice地址,如:http: //localhost/ASPxproject/WebServiceSolution/SiteBService/service1.asmx,你必 須保證你輸入的webservice存在.然後引用即可,注意:web引用名將作為你加入的webservice的名字空間.比如你輸入了:SiteA, 那服務的實例化將是這樣:SiteA.Service1 serviceA=new SiteA.Service1();(Service1是服務的類名).
完成了這一步,service的調用似乎變的那麼簡單,我們已經實現了遠程實例化,接下來的遠程調用也是一樣的容易.下面給出資源采集站ServiceGatherSite的綁定代碼(只采集A站點的信息)
以下是引用片段:
//綁定數據 public void BindData() { serviceA=new SiteA.Service1(); DataSet ds=new DataSet(); XmlNode xmlNode1; XmlDataDocument xd=new XmlDataDocument(); StringBuilder xmlString1; xmlNode1=serviceA.GetSiteAData(strSearch); if(xmlNode1==null) //--存儲過程執行失敗 return; xmlString1=new StringBuilder(xmlNode1.OuterXml); if(xmlString1.ToString().Equals("")) return ; xd.LoadXml(xmlString1.ToString()); ds.ReadXml(new XMLNodeReader(xd)); DataGrid1.DataSource=ds.Tables["list"].DefaultVIEw; DataGrid1.DataBind(); }
此段代碼給出了XML轉化成dataset的解決方案,雖然這不是必須的,但畢竟在ASP.Net裡,dataset占的作用之重,誰都知道的.其他的請 朋友們先看(呵呵,個中高手就免了),在下篇文章中將會有對它的一些解釋與及多服務分布調用的解決方案,寫了三篇了,發現似乎還是有些朋友看的,那我就獻 丑繼續寫下去好了,大家有什麼意見也希望提出,在下的理解存在偏駁也再所難免,希望諒解:)
接上篇文章,我們先簡單說明下綁定函數.首先實例化ServiceA,這個和一般類的實例化並沒有不同..接下來用xmlNode1來接受函數的返回值,接下來是構造XML,並將其轉化為dataset,這是通用的方法,如果是剛接觸不久的朋友,最好能記下這種方法.
接下來給出異步調用兩個服務的代碼
以下是引用片段:
//綁定數據 public void BindData() { IAsyncResult ar1; IAsyncResult ar2; serviceA=new SiteA.Service1(); serviceB=new SiteB.Service1(); DataSet ds=new DataSet(); XmlNode xmlNode1,xmlNode2; XmlDataDocument xd=new XmlDataDocument(); StringBuilder xmlString1,xmlString2; //--簡單的異步調用 ar1=serviceA.BeginGetSiteAData(strSearch,null,null); ar2=serviceB.BeginGetSiteAData(strSearch,null,null); xmlNode1=serviceA.EndGetSiteAData(ar1); xmlNode2=serviceB.EndGetSiteAData(ar2); //---------- if(xmlNode1==null&&xmlNode2==null) //--存儲過程執行失敗 return; xmlString1=new StringBuilder(xmlNode1.OuterXml); xmlString2=new StringBuilder(xmlNode2.OuterXml); xmlString1=MakeNewXmlString(xmlString1,xmlString2); //生成新的xml if(xmlString1.ToString().Equals("")) return ; xd.LoadXml(xmlString1.ToString()); ds.ReadXml(new XmlNodeReader(xd)); DataGrid1.DataSource=ds.Tables["list"].DefaultVIEw; DataGrid1.DataBind(); } //生成新XML public StringBuilder MakeNewXMLString(StringBuilder str1,StringBuilder str2) { str1=str1.Replace("",""); str2=str2.Replace("",""); str1.Append(str2.ToString()); return str1; }
這有兩個需要注意的地方,一個是XML構造,還有就是異步調用的實現,請讀者,自己理解
下面講下通過dll來引用webservice的方法,我只把流程介紹下.
首先,在IE輸入服務的地址,如:http://www.xxx.com/service.asmx
然後寫輸入http://www.xxx.com/service.asmx?wsdl
打開後,另存為xxx.wsdl
然後用vs的命令提示符來編譯:wsdl /namespace:SiteA ServiceA.wsdl
生成名字空間為sitea的代理類
最後csc /out:ServiceA.dll /t:library Service1.cs ,其中service1.cs為代理類文件
最後引用dll就可以了.