自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

詳解F#對象序列化為XML的實現(xiàn)方法

開發(fā) 后端
在這里我們將討論的是F#對象序列化為XML的實現(xiàn)方法,希望對大家用好F#開發(fā)有所幫助。

本文將從F#對象開始,詳細描述F#對象序列化為XML的實現(xiàn)方法,期間還與C#進行了對比。希望通過本文,能讓大家更好的理解F#。

#T#

這兩天在用F#寫一小段代碼,需要把一些對象存到外部文件中去。這個功能很容易,因為.NET本身就內(nèi)置了序列化功能。方便起見,我打算將這個F#對象序列化成XML而不是二進制數(shù)據(jù)流。這意味著我需要使用XmlSerializer而不是BinaryFormatter。這本應沒有問題,但是在使用時候還是發(fā)生了一些小插曲。

定義類型

在F#中有多種定義方式。除了F#特有的Record類型外,在F#中也可以定義普通的“類”,如:

 

  1. #light  
  2. module XmlSerialization  
  3. type Post() =  
  4.     [<DefaultValue>]  
  5.     val mutable Title : string 
  6.     [<DefaultValue>]  
  7.     val mutable Content : string 
  8.     [<DefaultValue>] 

val mutable Tags : string array上面的代碼在XmlSerialization模塊中定義了一個Post類,其中包含三個公開字段。簡單地說,它和C#中的如下定義等價:

 

  1. public class Post  
  2. {  
  3.     public string Title;  
  4.     public string Content;  
  5.     public string[] Tags;  

可見,在定義這種簡單類型時,F(xiàn)#并沒有什么優(yōu)勢,反而需要更多的代碼。

使用XmlSerializer進行序列化

原本我以為使用XmlSerializer來序列化一個對象非常容易,寫一個簡單的(泛型)函數(shù)就可以了:

  1. let byXmlSerializer (graph: 'a) =  
  2.     let serializer = new XmlSerializer(typeof<'a>)  
  3.     let writer = new StringWriter()  
  4.     serializer.Serialize(writer, graph)  
  5.     writer.ToString() 

使用起來更加不在話下:

 

  1. let post = new XmlSerialization.Post()  
  2. post.Title <- "Hello" 
  3. post.Content <- "World" 
  4. post.Tags <- [| "Hello""World" |]  
  5. let xml = XmlSerialization.byXmlSerializer(post) 

但是,在運行的時候,XmlSerializer的構造函數(shù)卻拋出了InvalidOperationException:

XmlSerialization cannot be serialized. Static types cannot be used as parameters or return types.
這句話的提示似乎是在說XmlSerialization是一個靜態(tài)類型——但這其實是F#的模塊啊。不過使用.NET Reflector查看編譯后的程序集便會發(fā)現(xiàn),其實Post類是這樣定義的:

  1. public static class XmlSerialization  
  2. {  
  3.     public class Post { ... }  

雖然.NET中也有“模塊”的概念,但是它和F#中的模塊從各方面來講幾乎沒有相同之處。F#的模塊會被編譯為靜態(tài)類,自然模塊中的方法或各種函數(shù)便成為靜態(tài)類中的內(nèi)嵌類型及方法。這本沒有問題,從理論上來說XmlSerializer也不該有問題,不是嗎?

可惜XmlSerializer的確有這樣的問題,我認為這是個Bug——但就算這是個Bug也無法解決目前的狀況。事實上,互聯(lián)網(wǎng)上也有人提出這個問題,可惜半年來都沒有人回應。

手動序列化

那么我又該怎么做呢?我想,算了,既然如此,我們進行手動序列化吧。反正就是簡單的對象,寫起來應該也不麻煩。例如在C#中我們便可以:

 

  1. public class Post  
  2. {  
  3.     ...  
  4.     public string ToXml()  
  5.     {  
  6.         var xml =   
  7.             new XElement("Post",  
  8.                 new XElement("Title"this.Title),  
  9.                 new XElement("Content"this.Content),  
  10.                 new XElement("Tags",  
  11.                     this.Tags.Select(t => new XElement("Tag", t))));  
  12.         return xml.ToString();  
  13.     }  

很簡單,不是嗎?但是用F#寫同樣的邏輯便有一些問題了,最終得到的結果是:

 

  1. type Post() =   
  2.     ...  
  3.  
  4.     member p.ToXml() =  
  5.         let xml = new XElement(XName.Get("Post"))  
  6.         xml.Add(new XElement(XName.Get("Title"), p.Title))  
  7.         xml.Add(new XElement(XName.Get("Content"), p.Content))  
  8.  
  9.         let tagElements = p.Tags |> Array.map (fun t -> new XElement(XName.Get("Tag"), t))  
  10.         xml.Add(new XElement(XName.Get("Tags"), tagElements))  
  11.           
  12.         xml.ToString() 

C#之所以可以寫的簡單,其中有諸多因素:

XElement的構造函數(shù)***使用了params object[],這意味著我們可以把參數(shù)“羅列”出來,而不需要顯式地構造一個數(shù)組。

XElement的構造函數(shù)接受的其實是XName類型參數(shù),但字符串可以被隱式地轉(zhuǎn)化為XName類型。

XElement的構造函數(shù)可以將IEnumerable<XElement>對象轉(zhuǎn)化為獨立的元素。 但是,除了***一條外,其他兩個特性在F#里都無法享受到。因此,我們只能用命令式編程的方式編寫此類代碼。您可以發(fā)現(xiàn),這樣的F#代碼幾乎可以被自動轉(zhuǎn)化為Java代碼。F#在寫這樣的代碼時實在沒有優(yōu)勢。

使用DataContractSerializer

手動進行XML序列化雖然并不困難,但是實在麻煩。這不是一種通用的做法,我們必須為每個類型各寫一套序列化(和反序列化)邏輯,在類型字段有所改變的時候,序列化和反序列化的邏輯還必須有所變化。就在我打算寫一個簡單的,通用的XML序列化方法時,我忽然想到以前看到過的一篇文章,說是在.NET 3.0中發(fā)布了新的類庫:DataContractSerializer。

DataContractSerializer看似和WCF有關,如DataContractAttribute,DataMemberAttribute等標記最典型的作用也一直用在WCF里。但事實上,這些類型都是定義在System.Runtime.Serialization.dll中的,這意味著這些功能從設計之初與WCF分離開來,可以獨立使用。那么我們不如嘗試一下吧:

  1. let serialize (graph : 'a) =   
  2.     let serializer = new DataContractSerializer(typeof<'a>)  
  3.     let textWriter = new StringWriter();  
  4.     let xmlWriter = new XmlTextWriter(textWriter);  
  5.     serializer.WriteObject(xmlWriter, graph)  
  6.     textWriter.ToString() 

果然好用,DataContractSerializer并沒有出現(xiàn)XmlSerializer那樣傻乎乎地錯誤。自然,與之相對的反序列化函數(shù)也很容易寫:

  1. let deserialize<'a> xml =   
  2.     let serializer = new DataContractSerializer(typeof<'a>)  
  3.     let textReader = new StringReader(xml)  
  4.     let xmlReader = new XmlTextReader(textReader)  
  5.     serializer.ReadObject(xmlReader) :?> 'a 

試驗一下,看看效果?

 

  1. let post = new XmlSerialization.Post()  
  2. post.Title <- "Hello" 
  3. post.Content <- "World" 
  4. post.Tags <- [| "Hello""World" |]  
  5.  
  6. let xml = XmlSerialization.serialize post  
  7. let post' = XmlSerialization.deserialize<XmlSerialization.Post> xml 

經(jīng)過更多試驗,我發(fā)現(xiàn)DataContractSerializer對于復雜類型的字段也可以正常應對,而得到這些功能也只需要在目標類型上標記一個SerializableAttribute就行了,更細節(jié)的控制也可以通過DataContractAttribute等進行控制。這樣看來,XmlSerializer似乎已經(jīng)可以退出歷史舞臺了?

原文標題:F#中的XML序列化

鏈接: http://www.cnblogs.com/JeffreyZhao/archive/2010/01/03/fsharp-xml-serialization.html

責任編輯:彭凡 來源: 博客園
相關推薦

2009-11-16 09:05:46

CodeTimer

2009-02-24 10:09:02

XMLJava對象

2009-09-09 15:47:27

XML序列化和反序列化

2012-04-13 10:45:59

XML

2009-09-09 14:45:41

XML序列化和反序列化

2010-04-07 16:51:59

F#

2011-06-01 15:05:02

序列化反序列化

2010-01-26 08:25:06

F#語法F#教程

2021-11-18 11:48:46

ObjectInputJava

2009-08-19 09:42:34

F#并行排序算法

2018-03-19 10:20:23

Java序列化反序列化

2011-05-18 15:20:13

XML

2009-06-14 22:01:27

Java對象序列化反序列化

2009-09-09 17:14:24

XML序列化

2011-06-01 14:26:11

序列化

2010-01-07 10:04:18

F#函數(shù)式編程

2009-08-06 11:16:25

C#序列化和反序列化

2010-01-08 13:25:07

ibmdwXML

2010-03-16 09:09:04

F#

2010-01-15 08:33:13

F#F#類型推斷F#教程
點贊
收藏

51CTO技術棧公眾號