准备写个WinCE平台与数据库服务器数据通讯交互方面的专题文章,今天先整理个Web Service通讯方式。
公司目前的硬件产品平台是WinCE5.0,数据通讯是连接服务器与终端的桥梁,关系着终端的数据能否准确及时高效抵达服务器,是整个项目成败的关键。原先公司有同事用VC写过一个程序用Socket进行数据通讯,但一直问题不断。年前我开始探索用SqlCE与SqlServer数据同步方式进行数据上传与下载,通讯已经正常稳定。这方面的文章后续再整理。
Web Service用于PC间通讯的文章网上有很多,但用于WinCE平台调用的经验总结并不多见。Web Service的程序编写与配置调用相对来讲比较简单,Visual Studio里直接新建一个“Asp.net web 服务应用程序”就可以创建一个web Service项目了。其中的代码根据实际需求编写就行,这方面就不详述了。
终端设备是通过GPRS来进行数据传输的,因此,数据流量是非常重要的问题,应当尽可能少的减少数据传输,流量可是Money,压缩技术是关键。Google大法,找到了一款物美价廉的东东-Ihttp://www.icsharpcode.net/OpenSource/SharpZipLib/Default.aspx 所谓物美是这款代码支持Dot net CF平台,所谓价廉是这款代码完全开源免费。
操刀开工。。。先建一个直接返回DataSet集的Web Service服务
private SqlConnection Conn; private string ConnString = "Data Source=(local);Initial Catalog=Northwind;uid=sa;pwd=sa;"; dataConnection#region dataConnection private DataSet GetNorthwindDataSet() { return ExecuteSql("select * from Employees"); } private DataSet ExecuteSql(string mysql) { DataSet dataSet = new DataSet(); SqlDataAdapter adapter = new SqlDataAdapter(mysql, this.Conn); try { if (this.Conn.State == ConnectionState.Closed) { this.Conn.Open(); } adapter.Fill(dataSet, "table"); } catch (Exception exception) { HttpContext.Current.Response.Write(exception.Message); HttpContext.Current.Response.End(); } finally { if ((this.Conn != null) && (this.Conn.State == ConnectionState.Open)) { this.Conn.Close(); } adapter.Dispose(); } return dataSet; } #endregion //方法一:直接返回 DataSet 对象 [WebMethod(Description = "直接返回 DataSet 对象。")] public DataSet GetDataSet() { DataSet dataSet = GetNorthwindDataSet(); return dataSet; }
建立一个智能设备应用程序,添加Web引用,我这里用的是静态引用,没有用动态引用的原因是,试过网上的动态生成WebService引用的代码,效率远比静态引用要低很多,考虑终端设备资源的有限性,还是用的静态引用。建立好项目后在界面上添加一个button和datagrid控件,添加代码:
private webSer.Service1 ws; private void FrmMain_Load(object sender, EventArgs e) { ws = new DeviceApplication1.webSer.Service1(); } //webmethod直接返回dataset private void btnDataSet_Click(object sender, EventArgs e) { try { this.dtGrid.DataSource = null; DateTime dtBegin = DateTime.Now; DataSet ds = ws.GetDataSet(); DateTime dtDown = DateTime.Now; this.dtGrid.DataSource = ds.Tables[0]; MessageBox.Show(string.Format("下载耗时:{0},绑定数据耗时:{1},数据量:{2}", dtDown - dtBegin, DateTime.Now - dtDown, ds.GetXml().Length)); } catch (Exception ex) { MessageBox.Show(ex.ToString()); } }
连接好终端设备,测试点击按钮,几秒后DataGrid表格正确显示数据,OK,说明WinCE已经能够正确调用Web Service了。如果不能正确调用,检察WebService发布与Web引用是否正确,数据库配置是否正确。
接下来就是要把数据进行压缩了,压缩前要进行数据的序列化,序列化代码如下:
/**//// <summary> /// 序列化数据成byte[] /// </summary> /// <param name="o">未序列化的数据</param> /// <returns>返回序列化后的byte[]数组</returns> public static byte[] byteXmlSerializer(object o) { if (o == null) { throw new ArgumentNullException("null input"); } try { System.Xml.Serialization.XmlSerializer ser = new System.Xml.Serialization.XmlSerializer(o.GetType()); System.IO.MemoryStream mem = new MemoryStream(); System.Xml.XmlTextWriter writer = new System.Xml.XmlTextWriter(mem, Encoding.Default); ser.Serialize(writer, o); writer.Close(); return mem.ToArray(); } catch (Exception ex) { throw ex; } } /**//// <summary> /// 反序列化数据 /// </summary> /// <param name="input">序列化后的数据</param> /// <param name="type">被序列化的数据类型</param> /// <returns>返回反序列化后的object数据</returns> public static object objXmlDeserialize(byte[] input, Type type) { if (input == null) { throw new ArgumentNullException("null input"); } try { XmlSerializer mySerializer = new XmlSerializer(type); StreamReader stmRead = new StreamReader(new MemoryStream(input), System.Text.Encoding.Default); return mySerializer.Deserialize(stmRead); } catch (Exception ex) { throw ex; } }
ICSharpCode提供了多种数据压缩的方式,我这里测试了四种方式:BZip,Deflate,GZip,Zip
private static int buffSize = 2048;//指定压缩块缓存的大小,一般为2048的倍数 /**//// <summary> /// BZIP2压缩数据 /// </summary> /// <param name="input">原始未压缩数据</param> /// <returns>压缩后的byte[]数据</returns> public static byte[] BZipCompress(byte[] input) { if (input == null) { throw new ArgumentNullException("null input"); } try { //int buffSize = 2048;//指定压缩块的大小,一般为2048的倍数 using (MemoryStream outmsStrm = new MemoryStream()) { using (MemoryStream inmsStrm = new MemoryStream(input)) { BZip2.Compress(inmsStrm, outmsStrm, buffSize); } return outmsStrm.ToArray(); } } catch (Exception ex) { throw ex; } } /**//// <summary> /// 解压缩BZIP2数据 /// </summary> /// <param name="input">被BZIP2压缩过的byte[]数据</param> /// <returns>解压后的byte[]数据</returns> public static byte[] BZipDeCompress(byte[] input) { if (input == null) { throw new ArgumentNullException("null input"); } try { using (MemoryStream outmsStrm = new MemoryStream()) { using (MemoryStream inmsStrm = new MemoryStream(input)) { BZip2.Decompress(inmsStrm, outmsStrm); } return outmsStrm.ToArray(); } } catch (Exception ex) { throw (ex); } } /**//// <summary> /// 压缩Deflater数据 /// </summary> /// <param name="input">待压缩byte[]数据</param> /// <returns>返回压缩后的byte[]</returns> public static byte[] DeflaterCompress(byte[] input) { if (input == null) { throw new ArgumentNullException("null input"); } try { Deflater mDeflater = new Deflater(Deflater.BEST_COMPRESSION); //int buffSize = 2048;//131072 buff size using (MemoryStream outmsStrm = new MemoryStream()) { using (DeflaterOutputStream mStream = new DeflaterOutputStream(outmsStrm, mDeflater, buffSize)) { mStream.Write(input, 0, input.Length); } return outmsStrm.ToArray(); } } catch (Exception ex) { throw (ex); } } /**//// <summary> /// 解压缩Deflater数据 /// </summary> /// <param name="input">压缩过的byte[]数据</param> /// <returns>解压后的byte[]数据</returns> public static byte[] DeflaterDeCompress(byte[] input) { if (input == null) { throw new ArgumentNullException("null input"); } try { Int32 mSize; //int buffSize = 2048; byte[] buff = new byte[buffSize]; using (MemoryStream outmsStrm = new MemoryStream()) { using (InflaterInputStream mStream = new InflaterInputStream(new MemoryStream(input))) { while (true) { mSize = mStream.Read(buff, 0, buff.Length); if (mSize > 0) { outmsStrm.Write(buff, 0, mSize); } else { break; } } } return outmsStrm.ToArray(); } } catch (Exception ex) { throw (ex); } } /**//// <summary> /// GZIP压缩 /// </summary> /// <param name="input">未压缩的数据</param> /// <returns>GZIP压缩后的数据</returns> public static byte[] GZipCompress(byte[] input) { if (input == null) { throw new ArgumentNullException("null input"); } try { using (MemoryStream outmsStrm = new MemoryStream()) { using (GZipOutputStream gzip = new GZipOutputStream(outmsStrm)) { gzip.Write(input, 0, input.Length); } return outmsStrm.ToArray(); } } catch (Exception ex) { throw (ex); } } /**//// <summary> /// GZIP解压缩 /// </summary> /// <param name="input">压缩过的数据</param> /// <returns>解压后的数据</returns> public static byte[] GZipDeCompress(byte[] input) { if (input == null) { throw new ArgumentNullException("null input"); } try { using (MemoryStream outmsStrm = new MemoryStream()) { using (GZipInputStream gzip = new GZipInputStream(new MemoryStream(input))) { Int32 mSize; //int buffSize = 2048; byte[] buff = new byte[buffSize]; while (true) { mSize = gzip.Read(buff, 0, buffSize); if (mSize > 0) { outmsStrm.Write(buff, 0, mSize); } else { break; } } } return outmsStrm.ToArray(); } } catch (Exception ex) { throw (ex); } } /**//// <summary> /// ZIP压缩数据 /// </summary> /// <param name="input">待压缩的数据</param> /// <returns>ZIP压缩后的数据</returns> public static byte[] ZipCompress(byte[] input) { if (input == null) { throw new ArgumentNullException("null input"); } try { using (MemoryStream outmsStrm = new MemoryStream()) { using (ZipOutputStream zipStrm = new ZipOutputStream(outmsStrm)) { ZipEntry zn = new ZipEntry("znName"); zipStrm.PutNextEntry(zn); zipStrm.Write(input, 0, input.Length); } return outmsStrm.ToArray(); } } catch (Exception ex) { throw (ex); } } /**//// <summary> /// ZIP解压缩数据 /// </summary> /// <param name="input">压缩过的数据</param> /// <returns>解压后的数据</returns> public static byte[] ZipDeCompress(byte[] input) { if (input == null) { throw new ArgumentNullException("null input"); } try { using (MemoryStream outmsStrm = new MemoryStream()) { using (ZipInputStream zipStrm = new ZipInputStream(new MemoryStream(input))) { Int32 mSize; //int buffSize = 2048; byte[] buff = new byte[buffSize]; ZipEntry zn = new ZipEntry("znName"); while ((zn = zipStrm.GetNextEntry()) != null) { while (true) { mSize = zipStrm.Read(buff, 0, buffSize); if (mSize > 0) { outmsStrm.Write(buff, 0, mSize); } else { break; } } } } return outmsStrm.ToArray(); } } catch (Exception ex) { throw (ex); } }
添加WebService服务
//方法二:返回 DataSet 对象用序列化后的byte[]字节数组 [WebMethod(Description = "返回 DataSet 对象用序列化后的byte[]字节数组。")] public byte[] GetDataSetBytes() { DataSet dataSet = GetNorthwindDataSet(); return ComZipClass.zip.byteXmlSerializer(dataSet); } //方法三返回 DataSet对象用序列化并Bzip压缩后的byte[]字节数组 [WebMethod(Description = "返回 DataSet对象用序列化并Bzip压缩后的byte[]字节数组。")] public byte[] GetBZiipCompress() { DataSet dataSet = GetNorthwindDataSet(); byte[] ser = ComZipClass.zip.byteXmlSerializer(dataSet); byte[] zip = ComZipClass.zip.BZipCompress(ser); return zip; } [WebMethod(Description = "返回Deflater压缩")] public byte[] GetDeflaterCompress() { DataSet dataSet = GetNorthwindDataSet(); byte[] ser = ComZipClass.zip.byteXmlSerializer(dataSet); //byte[] ser = ZipClass.zip.byteXmlSerializer(dataSet); byte[] zip = ComZipClass.zip.DeflaterCompress(ser); return zip; } [WebMethod(Description = "返回Gzip压缩")] public byte[] GetGZipCompress() { DataSet dataSet = GetNorthwindDataSet(); byte[] ser = ComZipClass.zip.byteXmlSerializer(dataSet); byte[] zip = ComZipClass.zip.GZipCompress(ser); return zip; } [WebMethod(Description = "返回Zip压缩")] public byte[] GetZipCompress() { DataSet dataSet = GetNorthwindDataSet(); byte[] ser = ComZipClass.zip.byteXmlSerializer(dataSet); byte[] zip = ComZipClass.zip.ZipCompress(ser); return zip; }
添加终端设备调用
//webmethod序列化数据 private void btnSerial_Click(object sender, EventArgs e) { try { this.dtGrid.DataSource = null; DataSet ds = new DataSet(); DateTime dtBegin = DateTime.Now; byte[] datas = ws.GetDataSetBytes(); DateTime dtDown = DateTime.Now; ds = (DataSet)ComZipClass.zip.objXmlDeserialize(datas, typeof(DataSet)); DateTime dtSerial = DateTime.Now; this.dtGrid.DataSource = ds.Tables[0]; MessageBox.Show(string.Format("下载耗时:{0},解压序列化数据耗时:{1},绑定数据耗时:{2},数据量:{3}", dtDown - dtBegin,dtSerial-dtDown, DateTime.Now - dtDown, datas.Length)); } catch (Exception ex) { MessageBox.Show(ex.ToString()); } } //序列化并压缩BZIP后数据 private void btnBZip_Click(object sender, EventArgs e) { try { this.dtGrid.DataSource = null; DataSet ds = new DataSet(); DateTime dtBegin = DateTime.Now; byte[] datas = ws.GetBZiipCompress(); DateTime dtDown = DateTime.Now; byte[] uzip = ComZipClass.zip.BZipDeCompress(datas); DateTime dtUnzip = DateTime.Now; ds = (DataSet)ComZipClass.zip.objXmlDeserialize(uzip, typeof(DataSet)); DateTime dtSerial = DateTime.Now; this.dtGrid.DataSource = ds.Tables[0]; MessageBox.Show(string.Format("下载耗时:{0},解压BZIP耗时:{1},解压序列化耗时:{2},绑定数据耗时:{3},数据量:{4}", dtDown - dtBegin, dtUnzip - dtDown, dtSerial - dtUnzip, DateTime.Now - dtSerial, datas.Length)); } catch (Exception ex) { MessageBox.Show(ex.ToString()); } } //序列化并压缩Deflate后数据 private void btnDeflater_Click(object sender, EventArgs e) { try { this.dtGrid.DataSource = null; DataSet ds = new DataSet(); DateTime dtBegin = DateTime.Now; byte[] datas = ws.GetDeflaterCompress(); DateTime dtDown = DateTime.Now; byte[] uzip = ComZipClass.zip.DeflaterDeCompress(datas); DateTime dtUnzip = DateTime.Now; ds = (DataSet)ComZipClass.zip.objXmlDeserialize(uzip, typeof(DataSet)); DateTime dtSerial = DateTime.Now; this.dtGrid.DataSource = ds.Tables[0]; MessageBox.Show(string.Format("下载耗时:{0},解压Deflater耗时:{1},解压序列化耗时:{2},绑定数据耗时:{3},数据量:{4}", dtDown - dtBegin, dtUnzip - dtDown, dtSerial - dtUnzip, DateTime.Now - dtSerial, datas.Length)); } catch (Exception ex) { MessageBox.Show(ex.ToString()); } } //序列化并压缩GZIP后数据 private void btnGZip_Click(object sender, EventArgs e) { try { this.dtGrid.DataSource = null; DataSet ds = new DataSet(); DateTime dtBegin = DateTime.Now; byte[] datas = ws.GetGZipCompress(); DateTime dtDown = DateTime.Now; byte[] uzip = ComZipClass.zip.GZipDeCompress(datas); DateTime dtUnzip = DateTime.Now; ds = (DataSet)ComZipClass.zip.objXmlDeserialize(uzip, typeof(DataSet)); DateTime dtSerial = DateTime.Now; this.dtGrid.DataSource = ds.Tables[0]; MessageBox.Show(string.Format("下载耗时:{0},解压GZIP耗时:{1},解压序列化耗时:{2},绑定数据耗时:{3},数据量:{4}", dtDown - dtBegin, dtUnzip - dtDown, dtSerial - dtUnzip, DateTime.Now - dtSerial, datas.Length)); } catch (Exception ex) { MessageBox.Show(ex.ToString()); } } //序列化并压缩ZIP后数据 private void btnZip_Click(object sender, EventArgs e) { try { this.dtGrid.DataSource = null; DataSet ds = new DataSet(); DateTime dtBegin = DateTime.Now; byte[] datas = ws.GetZipCompress(); DateTime dtDown = DateTime.Now; byte[] uzip = ComZipClass.zip.ZipDeCompress(datas); DateTime dtUnzip = DateTime.Now; ds = (DataSet)ComZipClass.zip.objXmlDeserialize(uzip, typeof(DataSet)); DateTime dtSerial = DateTime.Now; this.dtGrid.DataSource = ds.Tables[0]; MessageBox.Show(string.Format("下载耗时:{0},解压ZIP耗时:{1},解压序列化耗时:{2},绑定数据耗时:{3},数据量:{4}", dtDown - dtBegin, dtUnzip - dtDown, dtSerial - dtUnzip, DateTime.Now - dtSerial, datas.Length)); } catch (Exception ex) { MessageBox.Show(ex.ToString()); } }
几个测试压缩数据dataset :268128;serial:269199;bzip:94561;Deflater:107049;gzip:108276;zip:108408;