- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章浅谈.Net中的序列化和反序列化由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
序列化和反序列化相信大家都经常听到,也都会用, 然而有些人可能不知道:.net为什么要有这个东西以及.net frameword如何为我们实现这样的机制, 在这里我也是简单谈谈我对序列化和反序列化的一些理解.
1、什么序列化和反序列化 。
序列化通俗地讲就是将一个对象转换成一个字节流的过程,这样就可以轻松保存在磁盘文件或数据库中。反序列化是序列化的逆过程,就是将一个字节流转换回原来的对象的过程.
然而为什么需要序列化和反序列化这样的机制呢?这个问题也就涉及到序列化和反序列化的用途了, 。
对于序列化的主要用途有:
并且如果把对象序列化成内存中的字节流,就可以利用一些其他的技术来处理数据,例如,对数据进行加密和压缩等.
2、序列化和反序列简单使用 。
.net framework 提供二种序列化方式:
序列化和反序列化的简单使用:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
|
using
system;
using
system.io;
using
system.runtime.serialization.formatters.binary;
namespace
serializable
{
[serializable]
public
class
person
{
public
string
personname;
[nonserialized]
public
string
personheight;
private
int
personage;
public
int
personage
{
get
{
return
personage; }
set
{ personage = value; }
}
public
void
write()
{
console.writeline(
"person name: "
+personname);
console.writeline(
"person height: "
+personheight);
console.writeline(
"person age: "
+ personage);
}
}
class
program
{
static
void
main(
string
[] args)
{
person person =
new
person();
person.personname =
"jerry"
;
person.personheight =
"175cm"
;
person.personage = 22;
stream stream = serialize(person);
//为了演示,都重置
stream.position = 0;
person =
null
;
person = deserialize(stream);
person.write();
console.read();
}
private
static
memorystream serialize(person person)
{
memorystream stream =
new
memorystream();
// 构造二进制序列化格式器
binaryformatter binaryformatter =
new
binaryformatter();
// 告诉序列化器将对象序列化到一个流中
binaryformatter.serialize(stream, person);
return
stream;
}
private
static
person deserialize(stream stream)
{
binaryformatter binaryformatter =
new
binaryformatter();
return
(person)binaryformatter.deserialize(stream);
}
}
}
|
主要是调用system.runtime.serialization.formatters.binary命名空间下的binnaryformatter类来进行序列化和反序列化,调用反序列化后的结果截图:
从中可以看出除了标记nonserialized的其他成员都能序列化,注意这个属性只能应用于一个类型中的字段,而且会被派生类型继承.
soap 和xml 的序列化和反序列化和上面类似,只需要改下格式化器就可以了, 这里我就不列出来了.
3、控制序列化和反序列化 。
有两种方式来实现控制序列化和反序列化:
第一种方式实现控制序列化和反序列化代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
|
using
system;
using
system.io;
using
system.runtime.serialization;
using
system.runtime.serialization.formatters.binary;
namespace
controlserialization
{
[serializable]
public
class
circle
{
private
double
radius;
//半径
[nonserialized]
public
double
area;
//面积
public
circle(
double
inputradiu)
{
radius = inputradiu;
area = math.pi * radius * radius;
}
[ondeserialized]
private
void
ondeserialized(streamingcontext context)
{
area = math.pi * radius * radius;
}
public
void
write()
{
console.writeline(
"radius is: "
+ radius);
console.writeline(
"area is: "
+ area);
}
}
class
program
{
static
void
main(
string
[] args)
{
circle c =
new
circle(10);
memorystream stream =
new
memorystream();
binaryformatter formatter =
new
binaryformatter();
// 将对象序列化到内存流中,这里可以使用system.io.stream抽象类中派生的任何类型的一个对象, 这里我使用了 memorystream类型。
formatter.serialize(stream,c);
stream.position = 0;
c =
null
;
c = (circle)formatter.deserialize(stream);
c.write();
console.read();
}
}
}
|
运行结果为:
注意:如果注释掉 ondeserialized属性的话,area字段的值就是0了,因为area字段没有被序列化到流中.
在上面需要序列化的对象中,格式化器只会序列化对象的radius字段的值。area字段中的值不会序列化,因为该字段已经应用了nonserializedattribute属性,然后我们用circle c=new circle(10)这样代码构建一个circle对象时,在内部,area会设置一个约为314.159这样的值,这个对象序列化时,只有radius的字段的值(10)写入流中, 但当反序列化成一个circle对象时,它的area字段的值会初始化为0,而不是约314.159的一个值。为了解决这样的问题,所以自定义一个方法应用ondeserializedattribute属性。此时的执行过程为:每次反序列化类型的一个实例,格式化器都会检查类型中是否定义了 一个应用了该attribute的方法,如果是,就调用该方法,调用该方法时,所有可序列化的字段都会被正确设置。除了ondeserializedattribute这个定制attribute,system.runtime.serialization命名空间还定义了onserializingattribute,onserializedattribute和ondeserializingattribute这些定制属性.
实现iserializable接口方式控制序列化和反序列化代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
|
using
system;
using
system.io;
using
system.runtime.serialization;
using
system.runtime.serialization.formatters.binary;
using
system.security.permissions;
namespace
controlserilization2
{
[serializable]
public
class
myobject : iserializable
{
public
int
n1;
public
intn2;
[nonserialized]
public
string
str;
public
myobject()
{
}
protected
myobject(serializationinfo info, streamingcontext context)
{
n1 = info.getint32(
"i"
);
n2 = info.getint32(
"j"
);
str = info.getstring(
"k"
);
}
[securitypermissionattribute(securityaction.demand, serializationformatter =
true
)]
public
virtual
void
getobjectdata(serializationinfo info, streamingcontext context)
{
info.addvalue(
"i"
, n1);
info.addvalue(
"j"
, n2);
info.addvalue(
"k"
, str);
}
public
void
write()
{
console.writeline(
"n1 is: "
+ n1);
console.writeline(
"n2 is: "
+ n2);
console.writeline(
"str is: "
+ str);
}
}
class
program
{
static
void
main(
string
[] args)
{
myobject obj =
new
myobject();
obj.n1 = 2;
obj.n2 = 3;
obj.str =
"jeffy"
;
memorystream stream =
new
memorystream();
binaryformatter formatter =
new
binaryformatter();
// 将对象序列化到内存流中,这里可以使用system.io.stream抽象类中派生的任何类型的一个对象, 这里我使用了 memorystream类型。
formatter.serialize(stream, obj);
stream.position = 0;
obj =
null
;
obj = (myobject)formatter.deserialize(stream);
obj.write();
console.read();
}
}
}
|
结果为:
此时的执行过程为:当格式化器序列化对象时,会检查每个对象,如果发现一个对象的类型实现了iserializable接口,格式化器会忽视所有定制属性,改为构造一个新的system.runtime.serialization.serializationinfo对象,这个对象包含了要实际为对象序列化的值的集合。构造好并初始化好serializationinfo对象后,格式化器调用类型的getobjectdata方法,并向它传递对serializationinfo对象的引用,getobjectdata方法负责决定需要哪些信息来序列化对象,并将这些信息添加到serializationinfo对象中,通过调用addvalue方法来添加需要的每个数据,添加好所有必要的序列化信息后,会返回至格式化器,然后格式化器获取已经添加到serializationinfo对象中的所有值,并将它们都序列化到流中,当反序列化时,格式化器从流中提取一个对象时,会为新对象分配内存,最初,这个对象的所有字段都设为0或null,然后,格式化器检查类型是否实现了iserializable接口,如果存在这个接口, 格式化器就尝试调用一个特殊构造器,它的参数和getobjectdata方法的完全一致.
4、格式化器如何序列化和反序列化 。
从上面的分析中可以看出,进行序列化和反序列化主要是格式化器在工作的,然而下面就是要讲讲格式化器是如何序列化一个应用了 serializableattribute 属性的对象.
接下来是解释格式化器如何自动反序列化一个应用了 serializableattribute属性的对象.
public static object populateobjectmembers(object obj,memberinfo[] members,object[] data);这个方法遍历数组,将每个字段初始化成对应的值.
注:格式化如何序列化和反序列对象部分摘自clr via c#(第三版),写在这里可以让初学者进一步理解格式化器在序列化和反序列化过程中所做的工作.
写到这里这篇关于序列化和反序列的文章终于结束了, 希望对自己以后复习和园子里的朋友有帮助.
以上就是浅谈.net中的序列化和反序列化的详细内容,更多关于.net 序列化和反序列化的资料请关注我其它相关文章! 。
原文链接:https://www.kancloud.cn/wizardforcel/learning-hard-csharp/111555 。
最后此篇关于浅谈.Net中的序列化和反序列化的文章就讲到这里了,如果你想了解更多关于浅谈.Net中的序列化和反序列化的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我是 python 的新手。我试图找到我的文本的频率分布。这是代码, import nltk nltk.download() import os os.getcwd() text_file=open(
我对安卓 fragment 感到困惑。我知道内存 fragment 但无法理解什么是 android fragment 问题。虽然我发现很多定义,比如 Android fragmentation re
尝试对 WordPress 进行 dockerise 我发现了这个场景: 2个数据卷容器,一个用于数据库(bbdd),另一个用于wordpress文件(wordpress): sudo docker
这个问题已经有答案了: From the server is there a way to know that my page is being loaded in an Iframe (1 个回答)
我正在玩小型服务器,试图对运行在其上的服务进行docker化。为简化起见,假设我必须主要处理:Wordpress和另一项服务。 在Docker集线器上有许多用于Wordpress的图像,但是它们似乎都
我想要发生的是,当帐户成功创建后,提交的表单应该消失,并且应该出现一条消息(取决于注册的状态)。 如果成功,他们应该会看到一个简单的“谢谢。请检查您的电子邮件。” 如果不是,那么他们应该会看到一条适当
就是这样,我需要为客户添加一个唯一标识符。通过 strip 元数据。这就是我现在完全构建它的方式,但是我只有最后一部分告诉我用户购买了哪个包。 我试着看这里: Plans to stripe 代码在这
我有一个类将执行一些复杂的操作,涉及像这样的一些计算: public class ComplexAction { public void someAction(String parameter
这个问题已经有答案了: maven add a local classes directory to module's classpath (1 个回答) 已关闭10 年前。 我有一些不应更改的旧 E
我使用 fragment 已经有一段时间了,但我经常遇到一个让我烦恼的问题。 fragment 有时会相互吸引。现在,我设法为此隔离了一个用例,它是这样的: Add fragment A(也使用 ad
我的 html 中有一个 ol 列表,上面有行条纹。看起来行条纹是从数字后面开始的。有没有办法让行条纹从数字开始? 我已经包含了正在发生的事情的片段 h4:nth-child(even) {
如何仅使用 css 将附加图像 html 化? 如果用纯 css 做不到,那我怎么能至少用一个图像来做 最佳答案 这不是真正的问题,而是您希望我们为您编写代码。我建议您搜索“css breadcrum
以下是 Joshua 的 Effective Java 的摘录: If you do synchronize your class internally, you can use various te
在这里工作时,我们有一个框向业务合作伙伴提供 XML 提要。对我们的提要的请求是通过指定查询字符串参数和值来定制的。其中一些参数是必需的,但很多不是。 例如,我们要求所有请求都指定一个 GUID 来标
我有 3 个缓冲区,其中包含在 32 位处理器上运行的 R、G、B 位数据。 我需要按以下方式组合三个字节: R[0] = 0b r1r2r3r4r5r6r7r8 G[0] = 0b g1g2g3g4
我最近发现了关于如何使用 History.js、jQuery 和 ScrollTo 通过 HTML5 History API 对网站进行 Ajax 化的要点:https://github.com/br
我们有一个 Spring Boot 应用程序,由于集成需要,它变得越来越复杂——比如在你这样做之后发送一封电子邮件,或者在你之后广播一条 jms 消息等等。在寻找一些更高级别的抽象时,我遇到了 apa
我正在尝试首次实施Google Pay。我面临如何指定gateway和gatewayMarchantId的挑战。 我所拥有的是google console帐户,不知道在哪里可以找到此信息。 priva
昨天下午 3 点左右,我为两个想要从一个 Azure 帐户转移到另一个帐户的网站设置了 awverify 记录。到当天结束时,Azure 仍然不允许我添加域,所以我赌了一把,将域和 www 子域重新指
我正在使用terms facet在elasticsearch服务器中获取顶级terms。现在,我的标签"indian-government"不被视为一个标签。将其视为"indian" "governm
我是一名优秀的程序员,十分优秀!