人无信不立/2008-04-26 22:30
永春阁
专注于技术,切不可沉湎于技术
博客园
社区
首页
新随笔
联系
管理
订阅
随笔- 226 文章- 20 评论- 1275
C#强化系列文章七:序列化和反序列化
序列化和反序列化我们可能经常会听到,其实通俗一点的解释,序列化就是把一个对象保存到一个文件或数据库字段中去,反序列化就是在适当的时候把这个文件再转化成原来的对象使用。
我想最主要的作用有:
1、在进程下次启动时读取上次保存的对象的信息
2、在不同的AppDomain或进程之间传递数据
3、在分布式应用系统中传递数据
......
在C#中常见的序列化的方法主要也有三个:BinaryFormatter、SoapFormatter、XML序列化
本文就通过一个小例子主要说说这三种方法的具体使用和异同点
这个例子就是使用三种不同的方式把一个Book对象进行序列化和反序列化,当然这个Book类首先是可以被序列化的。至于怎么使一个类可以序列化可以参见:
C#强化系列文章一:ViewState使用兼谈序列化
Book类
using
System;
using
System.Collections;
using
System.Text;
namespace
SerializableTest
{
[Serializable]
public
class
Book
{
public
Book()
{
alBookReader
=
new
ArrayList();
}
public
string
strBookName;
[NonSerialized]
public
string
strBookPwd;
private
string
_bookID;
public
string
BookID
{
get
{
return
_bookID; }
set
{ _bookID
=
value; }
}
public
ArrayList alBookReader;
private
string
_bookPrice;
public
void
SetBookPrice(
string
price)
{
_bookPrice
=
price;
}
public
void
Write()
{
Console.WriteLine(
"
Book ID:
"
+
BookID);
Console.WriteLine(
"
Book Name:
"
+
strBookName);
Console.WriteLine(
"
Book Password:
"
+
strBookPwd);
Console.WriteLine(
"
Book Price:
"
+
_bookPrice);
Console.WriteLine(
"
Book Reader:
"
);
for
(
int
i
=
0
; i
<
alBookReader.Count; i
++
)
{
Console.WriteLine(alBookReader[i]);
}
}
}
}
这个类比较简单,就是定义了一些public字段和一个可读写的属性,一个private字段,一个标记为[NonSerialized]的字段,具体会在下面的例子中体现出来
一、BinaryFormatter序列化方式
1、序列化,就是给Book类赋值,然后进行序列化到一个文件中
Book book
=
new
Book();
book.BookID
=
"
1
"
;
book.alBookReader.Add(
"
gspring
"
);
book.alBookReader.Add(
"
永春
"
);
book.strBookName
=
"
C#强化
"
;
book.strBookPwd
=
"
*****
"
;
book.SetBookPrice(
"
50.00
"
);
BinarySerialize serialize
=
new
BinarySerialize();
serialize.Serialize(book);
2、反序列化
BinarySerialize serialize
=
new
BinarySerialize();
Book book
=
serialize.DeSerialize();
book.Write();
3、测试用的
BinarySerialize类
using
System;
using
System.Collections.Generic;
using
System.Text;
using
System.IO;
using
System.Runtime.Serialization.Formatters.Binary;
namespace
SerializableTest
{
public
class
BinarySerialize
{
string
strFile
=
"
c:\\book.data
"
;
public
void
Serialize(Book book)
{
using
(FileStream fs
=
new
FileStream(strFile, FileMode.Create))
{
BinaryFormatter formatter
=
new
BinaryFormatter();
formatter.Serialize(fs, book);
}
}
public
Book DeSerialize()
{
Book book;
using
(FileStream fs
=
new
FileStream(strFile, FileMode.Open))
{
BinaryFormatter formatter
=
new
BinaryFormatter();
book
=
(Book)formatter.Deserialize(fs);
}
return
book;
}
}
}
主要就是调用System.Runtime.Serialization.Formatters.Binary空间下的BinaryFormatter类进行序列化和反序列化,以缩略型二进制格式写到一个文件中去,速度比较快,而且写入后的文件已二进制保存有一定的保密效果。
调用反序列化后的截图如下:
也就是说除了标记为NonSerialized的其他所有成员都能序列化
二、SoapFormatter序列化方式
调用序列化和反序列化的方法和上面比较类似,我就不列出来了,主要就看看SoapSerialize类
SoapSerialize类
using
System;
using
System.Collections.Generic;
using
System.Text;
using
System.IO;
using
System.Runtime.Serialization.Formatters.Soap;
namespace
SerializableTest
{
public
class
SoapSerialize
{
string
strFile
=
"
c:\\book.soap
"
;
public
void
Serialize(Book book)
{
using
(FileStream fs
=
new
FileStream(strFile, FileMode.Create))
{
SoapFormatter formatter
=
new
SoapFormatter();
formatter.Serialize(fs, book);
}
}
public
Book DeSerialize()
{
Book book;
using
(FileStream fs
=
new
FileStream(strFile, FileMode.Open))
{
SoapFormatter formatter
=
new
SoapFormatter();
book
=
(Book)formatter.Deserialize(fs);
}
return
book;
}
}
}
主要就是调用System.Runtime.Serialization.Formatters.Soap空间下的SoapFormatter类进行序列化和反序列化,使用之前需要应用System.Runtime.Serialization.Formatters.Soap.dll(.net自带的)
序列化之后的文件是Soap格式的文件(
简单对象访问协议(Simple Object Access Protocol,SOAP),是一种轻量的、简单的、基于XML的协议,它被设计成在WEB上交换结构化的和固化的信息。 SOAP 可以和现存的许多因特网协议和格式结合使用,包括超文本传输协议(HTTP),简单邮件传输协议(SMTP),多用途网际邮件扩充协议(MIME)。它还支持从消息系统到远程过程调用(RPC)等大量的应用程序。SOAP使用基于XML的数据结构和超文本传输协议(HTTP)的组合定义了一个标准的方法来使用Internet上各种不同操作环境中的分布式对象。
)
调用反序列化之后的结果和方法一相同
三、XML序列化方式
调用序列化和反序列化的方法和上面比较类似,我就不列出来了,主要就看看XmlSerialize类
XmlSerialize类
using
System;
using
System.Collections.Generic;
using
System.Text;
using
System.IO;
using
System.Xml.Serialization;
namespace
SerializableTest
{
public
class
XmlSerialize
{
string
strFile
=
"
c:\\book.xml
"
;
public
void
Serialize(Book book)
{
using
(FileStream fs
=
new
FileStream(strFile, FileMode.Create))
{
XmlSerializer formatter
=
new
XmlSerializer(
typeof
(Book));
formatter.Serialize(fs, book);
}
}
public
Book DeSerialize()
{
Book book;
using
(FileStream fs
=
new
FileStream(strFile, FileMode.Open))
{
XmlSerializer formatter
=
new
XmlSerializer(
typeof
(Book));
book
=
(Book)formatter.Deserialize(fs);
}
return
book;
}
}
}
从这三个测试类我们可以看出来其实三种方法的调用方式都差不多,只是具体使用的类不同
xml序列化之后的文件就是一般的一个xml文件:
book.xml
<?
xml version="1.0"
?>
<
Book
xmlns:xsi
="http://www.w3.org/2001/XMLSchema-instance"
xmlns:xsd
="http://www.w3.org/2001/XMLSchema"
>
<
strBookName
>
C#强化
</
strBookName
>
<
strBookPwd
>
*****
</
strBookPwd
>
<
alBookReader
>
<
anyType
xsi:type
="xsd:string"
>
gspring
</
anyType
>
<
anyType
xsi:type
="xsd:string"
>
永春
</
anyType
>
</
alBookReader
>
<
BookID
>
1
</
BookID
>
</
Book
>
输出截图如下:
也就是说采用xml序列化的方式只能保存public的字段和可读写的属性,对于private等类型的字段不能进行序列化
关于循环引用:
比如在上面的例子Book类中加入如下一个属性:
public Book relationBook;
在调用序列化时使用如下方法:
Book book
=
new
Book();
book.BookID
=
"
1
"
;
book.alBookReader.Add(
"
gspring
"
);
book.alBookReader.Add(
"
永春
"
);
book.strBookName
=
"
C#强化
"
;
book.strBookPwd
=
"
*****
"
;
book.SetBookPrice(
"
50.00
"
);
Book book2
=
new
Book();
book2.BookID
=
"
2
"
;
book2.alBookReader.Add(
"
gspring
"
);
book2.alBookReader.Add(
"
永春
"
);
book2.strBookName
=
"
.NET强化
"
;
book2.strBookPwd
=
"
*****
"
;
book2.SetBookPrice(
"
40.00
"
);
book.relationBook
=
book2;
book2.relationBook
=
book;
BinarySerialize serialize
=
new
BinarySerialize();
serialize.Serialize(book);
这样就会出现循环引用的情况,对于BinarySerialize和SoapSerialize可以正常序列化(.NET内部进行处理了),对于XmlSerialize出现这种情况会报错:
"序列化类型 SerializableTest.Book 的对象时检测到循环引用。"
posted @ 2008-03-18 17:39
永春
阅读(3674)
评论(30)
编辑
收藏
所属分类:
.Net
发表评论
回复
引用
查看
#1楼
2008-03-18 17:51 |
BoyLee
沙发,学习了
回复
引用
#2楼
2008-03-18 17:54 |
糖冒鸡屎 [未注册用户]
学习了
回复
引用
查看
#3楼
2008-03-18 17:56 |
自由、创新、研究、探索……
还有WCF中引入的第4种序列化
回复
引用
查看
#4楼
[
楼主
]2008-03-18 18:04 |
永春
@自由、创新、研究、探索……
WCF还没仔细看过,有时间来看看
回复
引用
查看
#5楼
2008-03-18 19:06 |
chegan
很好,学习了
回复
引用
查看
#6楼
2008-03-18 22:14 |
生鱼片
支持
回复
引用
#7楼
2008-03-18 22:42 |
wu zb [未注册用户]
太好了今天刚刚要用这个
回复
引用
#8楼
2008-03-18 22:43 |
wu zb [未注册用户]
谢谢楼主
回复
引用
#9楼
2008-03-18 23:36 |
盛大 [未注册用户]
兄弟你是我努力的目标。支持你。好好学习了。
回复
引用
查看
#10楼
2008-03-18 23:45 |
静◆思◆学 ★ network
--引用--------------------------------------------------
wu zb: 谢谢楼主
--------------------------------------------------------
回复
引用
查看
#11楼
[
楼主
]2008-03-19 08:54 |
永春
@chegan
@生鱼片
@wu zb
@盛大
@静◆思◆学 ★ network
谢谢大家支持
回复
引用
查看
#12楼
2008-03-19 09:19 |
狼Robot
using (FileStream fs = new FileStream(strFile, FileMode.Open))
这样的代码不知道楼主用的多不多,我的代码里几乎没出现过这样的代码.
回复
引用
查看
#13楼
[
楼主
]2008-03-19 09:25 |
永春
@狼Robot
这样的代码主要是保证资源的释放,
也就是说不需要手工调用fs.Close();方法,会在using结束后自动且一定会调用
回复
引用
查看
#14楼
2008-03-19 09:44 |
狼Robot
我知道他的用处,我的意思是楼主的代码里,这样的代码多不多?我比较菜.呵呵.
回复
引用
查看
#15楼
[
楼主
]2008-03-19 09:49 |
永春
@狼Robot
-_-,我的代码里一般能用到的地方都会用的,还是蛮多的。这个应该也是微软推荐的用法吧
回复
引用
查看
#16楼
2008-03-19 09:56 |
不知如何爱你
學習,收藏!
回复
引用
查看
#17楼
2008-03-19 10:38 |
Klesh Wong
不说说循环引用的情况吗?
回复
引用
查看
#18楼
2008-03-19 12:52 |
Sam Lin
请问这序列化和反序列化一般可应用于哪里呢?
回复
引用
查看
#19楼
[
楼主
]2008-03-19 13:01 |
永春
@Klesh Wong
谢谢提醒,我已经把循环引用的情况加进去了
回复
引用
查看
#20楼
[
楼主
]2008-03-19 13:02 |
永春
@Sam Lin
我想最主要的作用有:
1、在进程下次启动时读取上次保存的对象的信息
2、在不同的AppDomain或进程之间传递数据
3、在分布式应用系统中传递数据
不过很多要根据实际情况来看的
回复
引用
查看
#21楼
2008-03-19 15:27 |
蓝天旭日
总结得比较经典!
支持!
回复
引用
#22楼
2008-03-20 00:33 |
路客 [未注册用户]
谢谢楼主,很有帮助!
回复
引用
查看
#23楼
2008-03-23 17:04 |
情缘
好!
回复
引用
#24楼
2008-07-06 17:56 |
傅军 [未注册用户]
请问博主,那么ViewState使用的是哪种序列化方式?是这三种的一种吗?还是自己实现了一套序列化的方法?:)
回复
引用
查看
#25楼
[
楼主
]2008-07-07 12:54 |
永春
@傅军
这个我没看过,应该是类似BinaryFormatter的方式
回复
引用
#26楼
2008-07-07 13:27 |
傅军 [未注册用户]
ViewState是Base64编码过的。
如果用Base64解码以后得到的是一个字符串,这个字符串好像和二进制、SOAP、XML的格式都不像。
就不知道这个序列化是怎么弄的了。:)
回复
引用
查看
#27楼
2008-07-31 11:00 |
小手冰凉
最好能把什么时候用序列化和反序列化介绍一下!
回复
引用
查看
#28楼
[
楼主
]2008-07-31 12:19 |
永春
@小手冰凉
我想最主要的作用有:
1、在进程下次启动时读取上次保存的对象的信息
2、在不同的AppDomain或进程之间传递数据
3、在分布式应用系统中传递数据
4、在viewstate保存数据时
5