序列化是将一个对象保存到存储介质上或者将对象进行转换使之能够在网络上传送的行为在一个对象被序列化之后你会想要将它反序列化也就是将数据重新转换为可用的对象的行为这种类型的功能被用在一个对象必须被从一个上下文封送到另一个上下文的时候例如当对象跨越App域的时候另外一个例子是Web服务——对象在服务器上被序列化通过网络被(封送或)发送到的客户端然后被反序列化成有用的对象
从二进制到XMLNET框架为开发人员提供了很多序列化选项甚至允许开发人员创建自己的序列化例程在本文中我将着重讨论XML的序列化并告诉你如何利用这一内置的功能
XML序列化
NET框架提供的一种形式的序列化是XML序列化在这种类型的序列化里对象状态被以XML格式保存这使得被序列化的对象能够被不同的系统取得并修改甚至是那些不是用NET编写的系统另外一个优势是被序列化的对象对于人来说是可读和可写的——因此更新对象的方式莫过于打开写字板更改其值
XML序列化常常被来远程控制项目和Web服务项目里虽然你可能会在别的地方发现它比如DataSet封送在与Xpath查询和Predicate方法一起使用的时候XML序列化能够被用来实现面向对象的数据库——我会在今后的文章里谈这方面的内容
使用XML序列化
在NET框架里利用内置的XML序列化方法相对较为容易你只需要熟悉一些类和属性就可以开始使用简单的XML序列化了
SystemXmlSerialization命名空间含有使用XML序列化所需要的类和功能这个命名空间应该被放在使用XML序列化的类的顶部的一个using命令里
XmlSerializer类提供将对象序列化和反序列化的功能
XmlIgnore属性告诉XmlSerializer类跳过你不希望序列化的成员
这个列表只不过让你对序列化有一个初步的了解还有更多的对象可以供你在使用XML序列化的时候使用
列表A是将Customer对象序列化的一个简单例子这个Customer对象在下载文件的示例应用程序里被定义
列表A
Customer customer = newCustomer();
customerFirstName = Zach;
customerLastName = Smith;
XmlSerializer serializer = newXmlSerializer(typeof(Customer));
StringWriter writer = newStringWriter();
serializerSerialize(writer customer);
ConsoleWriteLine(writerToString());
正如你看到的XML序列化是一个很简单的过程我们只需要创建一个需要被序列化的对象创建用于这种类型对象的XmlSerializer并调用XmlSerializerSerialize被序列化的对象然后被写到提供的Stream里——在本文中是一个叫做writer的StringWriter
如果你看看被序列化的对象你会注意到它很容易读懂下面就是被序列化的Customer对象的内容
<?xml version= encoding=utf?>
<Customer xmlns:xsi=instance xmlns:xsd=>
<Orders />
<LastName>Smith</LastName>
<FirstName>Zach</FirstName>
</Customer>
在这里你会看到Customer对象的属性被序列化也就是Order对象的列表如果你将这个输出与Customer类的定义比较一下你会注意到Income属性没有被列出来这是因为Income属性本身还带有一个XmlIgnore属性它会让XML序列化库在序列化的时候跳过Income属性在序列化过程中跳过属性的一个副作用是这个属性不会被反序列化所以会在反序列化的时候总是带有默认的值
Orders节点是空的因为这个客户没有下订单但是如果我们将含有订单的Customer对象序列化那么我们就会得到类似下面的输出
<?xml version= encoding=utf?>
<Customer xmlns:xsi=instance xmlns:xsd=>
<Orders>
<Order>
<Total></Total>
</Order>
<Order>
<Total></Total>
</Order>
</Orders>
<LastName>Smith</LastName>
<FirstName>Zach</FirstName>
</Customer>
要注意每个订单对象都被单独序列化并放在被序列化的Customer对象的Orders节点里如果你想要反序列化这个Customer对象那么相关联的Order对象也会被反序列化
注要被序列化或者反序列化的每个对象都必须有一个空的默认构造函数如果对象不实现一个空的构造函数那么就会引发异常
反序列化对象
将对象反序列化就和将它们序列化一样简单下面的代码(列表B)显示了如何将反序列化保存在文件里的对象
列表B
XmlSerializer serializer = new XmlSerializer(typeof(Customer));
FileStream file = FileOpenRead(fileToWrite);
Customer customer = (Customer)serializerDeserialize(file);
fileClose();
上面这段代码与前面的序列化代码的唯一不同之处在于我们在这里调用了XmlSerializerDeserialize这个方法会返回一个对象我们然后就必须转到将要使用的对象类型上
其他用法
在某些情况下通过实现XML序列化来保存和检索设置文件是很有用的例如你可以只用创建一个含有所需属性的类并将它序列化到磁盘上而不需要编写一个自定义的XML分析例程来寻找和分析设置文件这还允许你在需要的时候对文件进行手动编辑让你不需要自己编写任何XML分析代码这种方式可以用于任何类型的信息应用程序可能需要保存从一个执行到另一个执行的这种类型的信息
正如我先前所说的将XML序列化与Predicate方法(被编译的搜索)Xpath(特别搜索)以及XmlDocument(提供器)对象一起使用你可以创建一个面向对象的数据库这是完全可行的我已经在进行这样一个项目所以我会在以后的文章里探讨它背后所隐藏的实现和思想其目标是利用不超过行的C#代码就创建完全可实现的面向对象的数据库
发挥XML序列化的优势
既然你已经熟悉XML序列化了那么我希望你能够在自己的项目里找到它的用武之地从我个人来讲我通过对复杂的应用程序设置使用XML序列化节省了大量的时间这还不算上我应用程序的其他一些模块所以我希望本文能够不让你犯我曾经犯过的错误!