一、什么是WebService:
简单通俗来说,就是企业之间、网站之间通过Internet来访问并使用在线服务,一些数据,由于安全性问题,不能提供数据库给其他单位使用,这时候可以使 用WebService服务提供。
二、创建WebService
创建WebService之后,我们就可以在文件里写返回数据的方法了。
三、返回数据的四种形式
笔者水平有限,只列出这四种数据的返回形式:
(1)直接返回DataSet对象
(2)返回DataSet对象用Binary序列化后的字节数组
(3)返回DataSetSurrogate对象用Binary序列化后的 字节数组
(4)返回DataSetSurrogate对象用Binary序列化并Zip 压缩后的字节数组
理论上来说,网络传输字节与传输时间,应该是递减的,其中,(3)(4)种方法需要引用微软提供的开源组件 下载地址:http://support.microsoft.com/kb/829740/zh-cn
下面展示这四种返回数据的代码,其中(1)是其三种方法的根本,都要得到一个DataSet作为根本,然后来做各种转换压缩的操作:
1 |
[WebMethod(Description = "直接返回DataSet对象" )] |
2 |
public DataSet GetDataSet() |
4 |
string connStr = System.Configuration.ConfigurationManager.ConnectionStrings[ "conn" ].ToString(); |
5 |
SqlConnection conn = new SqlConnection(connStr); |
6 |
string sql = "select * from china_city" ; |
8 |
SqlDataAdapter sda = new SqlDataAdapter(sql, conn); |
9 |
DataSet ds = new DataSet( "China" ); |
15 |
[WebMethod(Description = "直接返回DataSet对象,并用Binary序列化后的字节数组" )] |
16 |
public byte [] GetDataSetBytes() |
18 |
DataSet ds = GetDataSet(); |
19 |
BinaryFormatter ser = new BinaryFormatter(); |
20 |
MemoryStream ms = new MemoryStream(); |
21 |
ser.Serialize(ms, ds); |
22 |
byte [] buffer = ms.ToArray(); |
26 |
[WebMethod(Description = "直接返回DataSetSurrogate对象,并用Binary序列化后的字节数组" )] |
27 |
public byte [] GetDataSetSurrogateBytes() |
29 |
DataSet ds = GetDataSet(); |
30 |
DataSetSurrogate dss = new DataSetSurrogate(ds); |
31 |
BinaryFormatter ser = new BinaryFormatter(); |
32 |
MemoryStream ms = new MemoryStream(); |
33 |
ser.Serialize(ms, dss); |
34 |
byte [] buffer = ms.ToArray(); |
39 |
[WebMethod(Description = "直接返回DataSetSurrogate对象,并用Binary序列化后并且ZIP压缩的字节数组" )] |
40 |
public byte [] GetDataSetSurrogateZipBytes() |
42 |
DataSet ds = GetDataSet(); |
43 |
DataSetSurrogate dss = new DataSetSurrogate(ds); |
44 |
BinaryFormatter ser = new BinaryFormatter(); |
45 |
MemoryStream ms = new MemoryStream(); |
46 |
ser.Serialize(ms, dss); |
47 |
byte [] buffer = ms.ToArray(); |
48 |
byte [] bufferZip = ComPress(buffer); |
52 |
public byte [] ComPress( byte [] data) |
56 |
MemoryStream ms = new MemoryStream(); |
57 |
Stream zipStream = null ; |
58 |
zipStream = new GZipStream(ms, CompressionMode.Compress, true ); |
59 |
zipStream.Write(data, 0, data.Length); |
62 |
byte [] compressed_data = new byte [ms.Length]; |
63 |
ms.Read(compressed_data, 0, int .Parse(ms.Length.ToString())); |
64 |
return compressed_data; |
我们可以在浏览器中查看下WebService的效果,如图,在这个页面中,有提供四个方法,这四个方法就是上述我们写的四个返回数据的方法了,点击方法即可返回相应的数据,这样,我们数据提供方的代码就可以写好了,接下来,我们写调用数据的方法!
四、调用数据
客户端WebService程序
1 |
private void button1_Click( object sender, EventArgs e) |
3 |
com.dzbsoft.www.Service1 ds = new com.dzbsoft.www.Service1(); |
4 |
DateTime dtBegin = DateTime.Now; |
5 |
DataSet dataSet = ds.GetNorthwindDataSet(); |
6 |
this .label1.Text = string .Format( "耗时:{0}" , DateTime.Now - dtBegin); |
9 |
private void button2_Click( object sender, EventArgs e) |
11 |
com.dzbsoft.www.Service1 ds = new com.dzbsoft.www.Service1(); |
12 |
DateTime dtBegin = DateTime.Now; |
13 |
byte [] buffer = ds.GetDataSetBytes(); |
14 |
BinaryFormatter ser = new BinaryFormatter(); |
15 |
DataSet dataSet = ser.Deserialize( new MemoryStream(buffer)) as DataSet; |
16 |
this .label2.Text = string .Format( "耗时:{0}" , DateTime.Now - dtBegin) + " " + buffer.Length; |
19 |
private void button3_Click( object sender, EventArgs e) |
21 |
com.dzbsoft.www.Service1 ds = new com.dzbsoft.www.Service1(); |
22 |
DateTime dtBegin = DateTime.Now; |
23 |
byte [] buffer = ds.GetDataSetSurrogateBytes(); |
24 |
BinaryFormatter ser = new BinaryFormatter(); |
25 |
DataSetSurrogate dss = ser.Deserialize( new MemoryStream(buffer)) as DataSetSurrogate; |
26 |
DataSet dataSet = dss.ConvertToDataSet(); |
27 |
this .label3.Text = string .Format( "耗时:{0}" , DateTime.Now - dtBegin) + " " + buffer.Length; |
30 |
private void button4_Click( object sender, EventArgs e) |
32 |
com.dzbsoft.www.Service1 ds = new com.dzbsoft.www.Service1(); |
33 |
DateTime dtBegin = DateTime.Now; |
34 |
byte [] zipBuffer = ds.GetDataSetSurrogateZipBytes(); |
35 |
byte [] buffer = UnZipClass.Decompress(zipBuffer); |
36 |
BinaryFormatter ser = new BinaryFormatter(); |
37 |
DataSetSurrogate dss = ser.Deserialize( new MemoryStream(buffer)) as DataSetSurrogate; |
38 |
DataSet dataSet = dss.ConvertToDataSet(); |
39 |
this .label4.Text = string .Format( "耗时:{0}" , DateTime.Now - dtBegin) + " " + zipBuffer.Length; |
42 |
private void binddata(DataSet dataSet) |
44 |
this .dataGridView1.DataSource = dataSet.Tables[0]; |
45 |
this .label5.Text = "共计:" + dataSet.Tables[0].Rows.Count + "条记录" ; |
在数据返回的方法中,我们使用了数据的压缩,所以,在调用方这边,需要进行解压,代码:
2 |
public static class UnZipClass |
4 |
public static byte [] Decompress( byte [] data) |
8 |
MemoryStream ms = new MemoryStream(data); |
9 |
Stream zipStream = null ; |
10 |
zipStream = new GZipStream(ms, CompressionMode.Decompress); |
11 |
byte [] dc_data = null ; |
12 |
dc_data = ExtractBytesFromStream(zipStream, data.Length); |
20 |
public static byte [] ExtractBytesFromStream(Stream zipStream, int dataBlock) |
23 |
int totalBytesRead = 0; |
28 |
Array.Resize( ref data, totalBytesRead + dataBlock + 1); |
29 |
int bytesRead = zipStream.Read(data, totalBytesRead, dataBlock); |
34 |
totalBytesRead += bytesRead; |
36 |
Array.Resize( ref data, totalBytesRead); |
在上例中,调用四个方法的效果是一样的,唯一不同的是,传输过程中,数据量大小和传输时间的差异。
============2=============
效率调用问题,所以,我回说说如何实现同步与异步调用 webservice,如果说得哪里不对或者不好的地方,欢迎大家评论指导。
首先,什么是同步,什么是异步呢?打个比方来说,小明和小 华,互相打架,小明打了小华3下之后,小华才能打回小明,这叫同步,如果,小华勇敢点,在小明打了第一下开始做出反击,也打回小明,这叫异步。 也就是说,只能等待另外一个作业进行完才能进行下一个操作的叫同步,在另外一个作业进行的同时也进行其他操作,叫异步。
先创建一个webservice
3 |
using System.Web.Services; |
4 |
using System.Web.Services.Protocols; |
7 |
[WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)] |
10 |
public class GetWebService : System.Web.Services.WebService |
14 |
public string HelloWorld() |
17 |
for ( long i = 0; i < 1000000000; i++) |
21 |
return " Hello World" ; |
webservice创建好了,新建一个winform项目,引入webservice,我在引入webservice的时候,差点被坑爹了,原来。VS里是提供
Add Service References 和 Add Web References
这两种,其实就是年代遗留下来的问题。web引用是2.0版本的,而服务引用是3.5版本的,微软为了保持向前兼容的特性,也保留了这两种方法,分别可以看这里
项目右键 添加服务引用,如果你用的是VS2008,菜单可能是添加web引用。
如果是本地做学习测试之用的,浏览器浏览你创建的webservice,得到URL,如果是使用网络上的webservice,这里则输入给予的URL地址,点击前往即可,
再看看左下角的高级按钮吗?点击高级吧!!
把生成异步操作(必须勾上,不然没有异步方法)勾上,生成消息合同也需勾上,看到左下角的添加WEB引用了吗?这就是基于.NET Framework2.0 的。点击确定即可完成引入webservice。
两种不同版本的引入webservice也将造成代码的不同,所以,为了说明这个问题,我们也把2.0的引入方法也说明一下。
2.0的引入方法更加简洁,如果你在看浪曦的webservice视频教程,肯定很熟悉这个界面。我个人也是比较喜欢这种方法的。
编写代码
1 |
localhost.GetWebService webservice = new localhost.GetWebService(); |
3 |
ServiceReference1.GetWebServiceSoapClient getWebService = new ServiceReference1.GetWebServiceSoapClient(); |
6 |
private void btnSyn_Click( object sender, EventArgs e) |
8 |
string res = webservice.HelloWorld(); |
9 |
this .textBox1.Text += "完成了" ; |
10 |
this .textBox1.Text += res + System.Environment.NewLine; |
14 |
private void btnAsyn_Click( object sender, EventArgs e) |
17 |
webservice.HelloWorldCompleted += new localhost.HelloWorldCompletedEventHandler(AsyncHelloWorldComplete); |
19 |
webservice.HelloWorldAsync(); |
20 |
this .textBox1.Text += "完成了" + System.Environment.NewLine; |
24 |
void AsyncHelloWorldComplete( object sender, localhost.HelloWorldCompletedEventArgs e) |
26 |
string res = e.Result; |
27 |
this .textBox1.Text += res + System.Environment.NewLine; |
代码说明:
1、HelloWorld方法:同步直接调用webservice的方法,返回结果时输出“成功了”加上返回的结果;
2、webservice.HelloWorldAsync() :开始异步调用webservice
3、HelloWorldCompleted是webservice为我们提供委托调用,意思是将操作完成时执行的操作给参数中的方法执行,本例给了AsyncHelloWorldComplete方法执行;
执行效果:运行本例程序,你会发现,同步调用方法中,“完成了”这句话会与执行结果“Hello World”一起输出,在webservice还没执行完成的时候,小华不会打小明;
而异步调用方法中,“完成了”这句话先是输出到文本框中,等了数秒之后,再显示“Hello World”。这就是同步与异步调用webservice的区别了
如果需要在WebForm中异步调用,需要在页面属性中设置可以异步:Async=”true”
============================3======================
在前两讲里,我已经向大家演示了如何使用WebService、同步, 异步调用WebService,而在实际开发过程中,可能会有多个WebService接口供你选择,而在程序执行过程中才决定使用哪一个 WebService的情况,而以前的情况往往是添加指定的web引用调用WebService,而这一讲中,会讲述动态调用WebService,也就是知道WebService的地址而不用使用添加引用的方法来调用WebService。
首先贴出整个架构的示意图(图片来自浪曦),其中ServiceHelper类包括下面所示的五个步骤。
首先实现ServiceHelper类,代码如下:
1 |
using System.Collections.Generic; |
6 |
using System.Configuration; |
8 |
using System.CodeDom.Compiler; |
10 |
using System.Web.Services; |
11 |
using System.Web.Services.Description; |
12 |
using Microsoft.CSharp; |
14 |
namespace InvokeWebService |
16 |
public static class WebServiceHelper |
25 |
public static object InvokeWebService(string url, string methodname, object[] args) |
27 |
return InvokeWebService(url, null , methodname, args); |
37 |
public static object InvokeWebService(string url, string classname, string methodname, object[] args) |
39 |
string @ namespace = "ServiceBase.WebService.DynamicWebLoad" ; |
40 |
if (string.IsNullOrEmpty(classname)) |
42 |
classname = WebServiceHelper.GetClassName(url); |
45 |
WebClient wc = new WebClient(); |
46 |
Stream stream = wc.OpenRead(url + "?WSDL" ); |
47 |
ServiceDescription sd = ServiceDescription.Read(stream); |
48 |
ServiceDescriptionImporter sdi = new ServiceDescriptionImporter(); |
49 |
sdi.AddServiceDescription(sd, "" , "" ); |
50 |
CodeNamespace cn = new CodeNamespace(@ namespace ); |
52 |
CodeCompileUnit ccu = new CodeCompileUnit(); |
53 |
ccu.Namespaces.Add(cn); |
55 |
CSharpCodeProvider csc = new CSharpCodeProvider(); |
56 |
ICodeCompiler icc = csc.CreateCompiler(); |
58 |
CompilerParameters cplist = new CompilerParameters(); |
59 |
cplist.GenerateExecutable = false ; |
60 |
cplist.GenerateInMemory = true ; |
61 |
cplist.ReferencedAssemblies.Add( "System.dll" ); |
62 |
cplist.ReferencedAssemblies.Add( "System.XML.dll" ); |
63 |
cplist.ReferencedAssemblies.Add( "System.Web.Services.dll" ); |
64 |
cplist.ReferencedAssemblies.Add( "System.Data.dll" ); |
66 |
CompilerResults cr = icc.CompileAssemblyFromDom(cplist, ccu); |
67 |
if ( true == cr.Errors.HasErrors) |
69 |
System.Text.StringBuilder sb = new StringBuilder(); |
70 |
foreach (CompilerError ce in cr.Errors) |
72 |
sb.Append(ce.ToString() + System.Environment.NewLine); |
74 |
throw new Exception(sb.ToString()); |
77 |
System.Reflection.Assembly assembly = cr.CompiledAssembly; |
78 |
Type t = assembly.GetType(@ namespace + "." + classname, true , true ); |
79 |
object obj = Activator.CreateInstance(t); |
80 |
System.Reflection.MethodInfo mi = t.GetMethod(methodname); |
81 |
return mi.Invoke(obj, args); |
89 |
private static string GetClassName(string url) |
91 |
string[] parts = url.Split( '/' ); |
92 |
string[] pps = parts[parts.Length - 1 ].Split( '.' ); |
然后,我们可以新建1个WebService,看看是如何动态调用的:
1 |
private void button1_Click( object sender, EventArgs e) |
4 |
object b = InvokeWebService.WebServiceHelper.InvokeWebService(url, "HelloWorld" , null ); |
5 |
MessageBox.Show(b.ToString()); |
现在,整个项目中,没有像以往一样使用添加web引用来调用WebService,而是把WebService的调用地址,写在程序里面,结合业务逻辑可以动态调用wbeservice
ps:项目我是按着浪曦然后自己写的,WebServiceHelper类里面有些地方还不是很清楚,这里留下一份代码以作记录。