實現ADO.NET對象串行方法揭秘
學習ADO.NET時,你可能會遇到ADO.NET對象問題,這里將介紹解決方法,在這里拿出來和大家分享一下。只有兩個ADO.NET對象是被標記為可串行化的——DataTable和DataSet。.net Framework中的串行化是通過formatter對象來完成的,它們可以將一個對象實例保存到一個二進制或一個SOAP流(stream)中。.NET formatter用Reflection來提取任何必要的信息。然而,如果這個類實現了ISerializable接口,那么.NET formatter就會給接口的方法讓步,讓它們負責拷貝需要串行化到一個內存緩沖器中的所有的信息。DataTable和DataSet類都通過 ISerializable接口支持串行化。
如果你將一個DataTable或一個DataSet串行到一個二進制(binary stream)中,你應該可以得到非常緊湊的輸出結果。雖然你得到的結果文件是最小的,但遺憾的是,它實際上并不小。荒謬的是,你保存到一個二進制的DataSet比你用WriteXML方法保存到XML的同樣的DataSet要大很多。
要解釋這種情況,我們需要來看看ADO.NET對象是用什么方式被串行起來的。在串行一個DataSet對象時,它將基于XML的 DiffGram表示法保存在formatter的緩沖器中。在串行一個DataTable時,它首先創建了一個臨時的DataSet對象,將它定義為它的parent,然后作為一個DiffGram串行起來。
一個DiffGram是一個XML流,它提供了一個DataSet中表和行的有狀態的表示法。一個DiffGram文件是很詳細的,有些冗長。 DiffGram包含當前的數據,以及被修改的行和未解決的錯誤的初始值。當我們保存一個DataSet或一個DataTable時,所有這些信息就會被傳遞給serializer。被串行化的對象總是包含XML數據,因此即使當輸出流是二進制的時,***的輸出結果仍然會很大。
#T#你可以創建一個繼承DataTable或DataSet的新的可串行化的類來解決這個問題,并且更有效地保存ADO.NET對象。你必須用<Serizlizable()>屬性來標記新類,即使父類是可以串行化的。實際上,串行性(serizlizability)并不是一個可以自動繼承的類屬性。你從DataTable或DataSet構建的新類也可以實現ISerializable接口。當然,你可以為新類選擇一個不同的串行化方案。一個簡單而有效的方法就是將DataTable類的所有成員映射到數組和值成員中(見列表1)。
運用一個派生的類和一個自定義的串行化方案可以為一個DataSet對象節省多達80%的磁盤空間。節省的空間的比率取決于DataSet中的數據類型。你的數據越基于文本,節省的空間越多。然而,運用二進制的BLOB字段只可以節省大約25%的空間(下載一個完整的例子)。