WebService開發

1、什麼是WebService: web

簡單通俗來講,就是企業之間、網站之間經過Internet來訪問並使用在線服務,一些數據,因爲安全性問題,不能提供數據庫給其餘單位使用,這時候可使   用WebService服務提供。 sql

2、建立WebService 數據庫

564cf55f-d1fc-4dd7-8aad-f0514e96b018

建立WebService以後,咱們就能夠在文件裏寫返回數據的方法了。 數組

3、返回數據的四種形式 瀏覽器

筆者水平有限,只列出這四種數據的返回形式: 安全

(1)直接返回DataSet對象
(2)返回DataSet對象用Binary序列化後的字節數組
(3)返回DataSetSurrogate對象用Binary序列化後的 字節數組
(4)返回DataSetSurrogate對象用Binary序列化並Zip 壓縮後的字節數組 網絡

理論上來講,網絡傳輸字節與傳輸時間,應該是遞減的,其中,(3)(4)種方法須要引用微軟提供的開源組件  下載地址:http://support.microsoft.com/kb/829740/zh-cn 架構

下面展現這四種返回數據的代碼,其中(1)是其三種方法的根本,都要獲得一個DataSet做爲根本,而後來作各類轉換壓縮的操做:異步

ide

[WebMethod(Description = "直接返回DataSet對象")]public DataSet GetDataSet(){string connStr = System.Configuration.ConfigurationManager.ConnectionStrings["conn"].ToString();SqlConnection conn = new SqlConnection(connStr);string sql = "select * from china_city";conn.Open();SqlDataAdapter sda = new SqlDataAdapter(sql, conn);DataSet ds = new DataSet("China");sda.Fill(ds);conn.Close();return ds;} [WebMethod(Description = "直接返回DataSet對象,並用Binary序列化後的字節數組")]public byte[] GetDataSetBytes(){DataSet ds = GetDataSet();BinaryFormatter ser = new BinaryFormatter(); //序列化對象MemoryStream ms = new MemoryStream(); //內存流ser.Serialize(ms, ds);byte[] buffer = ms.ToArray(); //字節流return buffer;} [WebMethod(Description = "直接返回DataSetSurrogate對象,並用Binary序列化後的字節數組")]public byte[] GetDataSetSurrogateBytes(){DataSet ds = GetDataSet();DataSetSurrogate dss = new DataSetSurrogate(ds);BinaryFormatter ser = new BinaryFormatter(); //序列化對象MemoryStream ms = new MemoryStream(); //內存流ser.Serialize(ms, dss);byte[] buffer = ms.ToArray(); //字節流return buffer; } [WebMethod(Description = "直接返回DataSetSurrogate對象,並用Binary序列化後而且ZIP壓縮的字節數組")]public byte[] GetDataSetSurrogateZipBytes(){DataSet ds = GetDataSet();DataSetSurrogate dss = new DataSetSurrogate(ds);BinaryFormatter ser = new BinaryFormatter(); //序列化對象MemoryStream ms = new MemoryStream(); //內存流ser.Serialize(ms, dss);byte[] buffer = ms.ToArray(); //字節流byte[] bufferZip = ComPress(buffer);return buffer;}//壓縮方法public byte[] ComPress(byte[] data){try{MemoryStream ms = new MemoryStream();Stream zipStream = null;zipStream = new GZipStream(ms, CompressionMode.Compress, true);zipStream.Write(data, 0, data.Length);zipStream.Close();ms.Position = 0;byte[] compressed_data = new byte[ms.Length];ms.Read(compressed_data, 0, int.Parse(ms.Length.ToString()));return compressed_data;}catch{return null;}}

咱們能夠在瀏覽器中查看下WebService的效果,如圖,在這個頁面中,有提供四個方法,這四個方法就是上述咱們寫的四個返回數據的方法了,點擊方法便可返回相應的數據,這樣,咱們數據提供方的代碼就能夠寫好了,接下來,咱們寫調用數據的方法!

c4493088-61d7-4741-a81f-c2886b8cd663

4、調用數據

客戶端WebService程序

private void button1_Click(object sender, EventArgs e){com.dzbsoft.www.Service1 ds = new com.dzbsoft.www.Service1(); //new出WebService對象DateTime dtBegin = DateTime.Now;DataSet dataSet = ds.GetNorthwindDataSet();this.label1.Text = string.Format("耗時:{0}", DateTime.Now - dtBegin);binddata(dataSet);}private void button2_Click(object sender, EventArgs e){com.dzbsoft.www.Service1 ds = new com.dzbsoft.www.Service1();DateTime dtBegin = DateTime.Now;byte[] buffer = ds.GetDataSetBytes();BinaryFormatter ser = new BinaryFormatter();DataSet dataSet = ser.Deserialize(new MemoryStream(buffer)) as DataSet;this.label2.Text = string.Format("耗時:{0}", DateTime.Now - dtBegin) + " " + buffer.Length;binddata(dataSet);}private void button3_Click(object sender, EventArgs e){com.dzbsoft.www.Service1 ds = new com.dzbsoft.www.Service1();DateTime dtBegin = DateTime.Now;byte[] buffer = ds.GetDataSetSurrogateBytes();BinaryFormatter ser = new BinaryFormatter();DataSetSurrogate dss = ser.Deserialize(new MemoryStream(buffer)) as DataSetSurrogate;DataSet dataSet = dss.ConvertToDataSet();this.label3.Text = string.Format("耗時:{0}", DateTime.Now - dtBegin) + " " + buffer.Length;binddata(dataSet);}private void button4_Click(object sender, EventArgs e){com.dzbsoft.www.Service1 ds = new com.dzbsoft.www.Service1();DateTime dtBegin = DateTime.Now;byte[] zipBuffer = ds.GetDataSetSurrogateZipBytes();byte[] buffer = UnZipClass.Decompress(zipBuffer);BinaryFormatter ser = new BinaryFormatter();DataSetSurrogate dss = ser.Deserialize(new MemoryStream(buffer)) as DataSetSurrogate;DataSet dataSet = dss.ConvertToDataSet();this.label4.Text = string.Format("耗時:{0}", DateTime.Now - dtBegin) + " " + zipBuffer.Length;binddata(dataSet);}private void binddata(DataSet dataSet){this.dataGridView1.DataSource = dataSet.Tables[0];this.label5.Text = "共計:" + dataSet.Tables[0].Rows.Count + "條記錄";}

在數據返回的方法中,咱們使用了數據的壓縮,因此,在調用方這邊,須要進行解壓,代碼:

客戶端UnZipClass程序

public static class UnZipClass{public static byte[] Decompress(byte[] data){try{MemoryStream ms = new MemoryStream(data);Stream zipStream = null;zipStream = new GZipStream(ms, CompressionMode.Decompress);byte[] dc_data = null;dc_data = ExtractBytesFromStream(zipStream, data.Length);return dc_data;}catch{return null;}}public static byte[] ExtractBytesFromStream(Stream zipStream, int dataBlock){byte[] data = null;int totalBytesRead = 0;try{while (true){Array.Resize(ref data, totalBytesRead + dataBlock + 1);int bytesRead = zipStream.Read(data, totalBytesRead, dataBlock);if (bytesRead == 0){break;}totalBytesRead += bytesRead;}Array.Resize(ref data, totalBytesRead);return data;}catch{return null;}}}

在上例中,調用四個方法的效果是同樣的,惟一不一樣的是,傳輸過程當中,數據量大小和傳輸時間的差別。

效率調用問題,因此,我回說說如何實現同步與異步調用 webservice,若是說得哪裏不對或者很差的地方,歡迎你們評論指導。

首先,什麼是同步,什麼是異步呢?打個比方來講,小明和小 華,互相打架,小明打了小華3下以後,小華才能打回小明,這叫同步,若是,小華勇敢點,在小明打了第一下開始作出反擊,也打回小明,這叫異步。 也就是說,只能等待另一個做業進行完才能進行下一個操做的叫同步,在另一個做業進行的同時也進行其餘操做,叫異步。

先建立一個webservice

using System;using System.Web;using System.Web.Services;using System.Web.Services.Protocols;[WebService(Namespace = "http://tempuri.org/")][WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]//若要容許使用 ASP.NET AJAX 從腳本中調用此 Web 服務,請取消對下行的註釋。// [System.Web.Script.Services.ScriptService]public class GetWebService : System.Web.Services.WebService{[WebMethod]public string HelloWorld(){int res = 0;for (long i = 0; i < 1000000000; i++) //循環10億次,目的是模仿大批量操做,這裏至少須要數秒的操做以便看出異步的效果{res++;}return " Hello World";}}

webservice建立好了,新建一個winform項目,引入webservice,我在引入webservice的時候,差點被坑爹了,原來。VS裏是提供

Add Service References 和 Add Web References

這兩種,其實就是年代遺留下來的問題。web引用是2.0版本的,而服務引用是3.5版本的,微軟爲了保持向前兼容的特性,也保留了這兩種方法,分別能夠看這裏

5ec69403-f281-440b-ae25-008b42e059cb

項目右鍵 添加服務引用,若是你用的是VS2008,菜單多是添加web引用。

abe8a4a9-bdd0-4001-b57c-88a9c88b80b5

若是是本地作學習測試之用的,瀏覽器瀏覽你建立的webservice,獲得URL,若是是使用網絡上的webservice,這裏則輸入給予的URL地址,點擊前往便可,

再看看左下角的高級按鈕嗎?點擊高級吧!!

16a14bd6-136b-4f9a-a8a4-d36c34c325d3

把生成異步操做(必須勾上,否則沒有異步方法)勾上,生成消息合同也需勾上,看到左下角的添加WEB引用了嗎?這就是基於.NET Framework2.0 的。點擊肯定便可完成引入webservice。

兩種不一樣版本的引入webservice也將形成代碼的不一樣,因此,爲了說明這個問題,咱們也把2.0的引入方法也說明一下。

1163c10d-6179-4280-a9fc-865652917741

2.0的引入方法更加簡潔,若是你在看浪曦的webservice視頻教程,確定很熟悉這個界面。我我的也是比較喜歡這種方法的。

編寫代碼

localhost.GetWebService webservice = new localhost.GetWebService(); //經過2.0的添加WEB引用須要這種方式new出webservice對象 ServiceReference1.GetWebServiceSoapClient getWebService = new ServiceReference1.GetWebServiceSoapClient();//經過添加服務引用須要這種方式new出webservice對象 //同步調用webserviceprivate void btnSyn_Click(object sender, EventArgs e){string res = webservice.HelloWorld();this.textBox1.Text += "完成了";this.textBox1.Text += res + System.Environment.NewLine;} //異步調用webserviceprivate void btnAsyn_Click(object sender, EventArgs e){//給HelloWorld方法註冊調用完成時執行的方法AsyncHelloWorldCompletewebservice.HelloWorldCompleted += new localhost.HelloWorldCompletedEventHandler(AsyncHelloWorldComplete);//開始異步調用webservice.HelloWorldAsync();this.textBox1.Text += "完成了" + System.Environment.NewLine;} //完成webservice操做時會執行的方法void AsyncHelloWorldComplete(object sender, localhost.HelloWorldCompletedEventArgs e){string res = e.Result;this.textBox1.Text += res + System.Environment.NewLine;}

代碼說明:

86534613-b9b1-408b-abc3-9280eaf5e46f

Completed是webservice爲咱們提供委託調用,意思是將操做完成時執行的操做給參數中的方法執行,本例給了AsyncHelloWorldComplete方法執行;

執行效果:運行本例程序,你會發現,同步調用方法中,「完成了」這句話會與執行結果「Hello World」一塊兒輸出,在webservice還沒執行完成的時候,小華不會打小明;
而異步調用方法中,「完成了」這句話先是輸出到文本框中,等了數秒以後,再顯示「Hello World」。這就是同步與異步調用webservice的區別了

若是須要在WebForm中異步調用,須要在頁面屬性中設置能夠異步:Async=」true」

============================3======================

在前兩講裏,我已經向你們演示瞭如何使用WebService、同步, 異步調用WebService,而在實際開發過程當中,可能會有多個WebService接口供你選擇,而在程序執行過程當中才決定使用哪個 WebService的狀況,而之前的狀況每每是添加指定的web引用調用WebService,而這一講中,會講述動態調用WebService,也就是知道WebService的地址而不用使用添加引用的方法來調用WebService。

 

首先貼出整個架構的示意圖(圖片來自浪曦),其中ServiceHelper類包括下面所示的五個步驟。

0d1287af-0df1-44bd-b9c9-7f8783492eb8

using System.Collections.Generic;using System.Linq;using System.Text;using System.IO;using System.Configuration;using System.CodeDom;using System.CodeDom.Compiler;using System.Net;using System.Web.Services;using System.Web.Services.Description;using Microsoft.CSharp;namespace InvokeWebService{public static class WebServiceHelper{/// <summary>/// 動態調用WebService/// </summary>/// <param name="url">WebService地址</param>/// <param name="methodname">WebService方法名</param>/// <param name="args">參數列表</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">WebService方法名</param>/// <param name="args">參數列表</param>/// <returns>返回object</returns>public static object InvokeWebService(string url, string classname, string methodname, object[] args){string @namespace = "ServiceBase.WebService.DynamicWebLoad";if (string.IsNullOrEmpty(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();ICodeCompiler icc = csc.CreateCompiler();//設定編譯器的參數CompilerParameters cplist = new CompilerParameters();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);if (true == cr.Errors.HasErrors){System.Text.StringBuilder sb = new StringBuilder();foreach (CompilerError ce in cr.Errors){sb.Append(ce.ToString() + System.Environment.NewLine);}throw new Exception(sb.ToString());}// 生成代理實例並調用方法System.Reflection.Assembly assembly = cr.CompiledAssembly;Type t = assembly.GetType(@namespace + "." + classname, true, true);object obj = Activator.CreateInstance(t);System.Reflection.MethodInfo mi = t.GetMethod(methodname);return mi.Invoke(obj, args);}/// <summary>/// 獲得URL中的WebService名稱/// </summary>/// <param name="url">URL地址</param>/// <returns>如http://wwww.baidu.com/service.asmx 則返回service</returns>private static string GetClassName(string url){string[] parts = url.Split('/');string[] pps = parts[parts.Length - 1].Split('.');return pps[0];}}}

而後,咱們能夠新建1個WebService,看看是如何動態調用的:

private void button1_Click(object sender, EventArgs e){string url = "http://localhost:2697/Service1.asmx"; //用於作測試的WebServiceobject b = InvokeWebService.WebServiceHelper.InvokeWebService(url, "HelloWorld", null);MessageBox.Show(b.ToString());}

如今,整個項目中,沒有像以往同樣使用添加web引用來調用WebService,而是把WebService的調用地址,寫在程序裏面,結合業務邏輯能夠動態調用wbeservice

ps:項目我是按着浪曦而後本身寫的,WebServiceHelper類裏面有些地方還不是很清楚,這裏留下一份代碼以做記錄。

WSDL方式調用 webservice

具體步驟:

WSDL生成指定地址的cs文件

開始--》programes--》vs2008--》tools--》vs2008 command prompt 打開命令行窗口

wsdl  /l:cs /n:mynamespace /out:myservice.cs http://localhost/cmdwebservice/computeservice.asmx

而後去 命令窗口所在目錄找 myservice.cs這個文件

個人是在 D:\program files\Microsoft Visual Studio 9.0\VC\myservice.cs

而後copy myservice。cs 文件到 項目中的一個新的文件夾。修改namespace 與當前項目名稱一致, 以使得當前目錄可以順利調用

相關文章
相關標籤/搜索