C#通过序列化实现深拷贝

简介: 关于浅拷贝和深拷贝,我的理解不是很清晰,简单说来,我认为是这样子:   浅拷贝:引用成员在被拷贝时仅复制源对象中引用成员的地址到新对象中,所以在新对象中对引用成员进行更改会影响到源对象(除对引用成员进行赋值外)。
关于 浅拷贝深拷贝,我的理解不是很清晰,简单说来,我认为是这样子:

  浅拷贝:引用成员在被拷贝时仅复制源对象中引用成员的地址到新对象中,所以在新对象中对引用成员进行更改会影响到源对象(除对引用成员进行赋值外)。

  深拷贝:引用成员在拷贝时新建一个引用对象到新对象中,且将源对象中引用对象的成员值复制到新对象的引用对象中,所以在新对象中对引用成员进行更改不会影响源对象。

  说起来概念也简单,我想大家纠结的是如何实现深拷贝?浅拷贝的实现很简单,调用Object.MemberwiseClone就万事大吉了。在网上找了一个通过序列化实现深拷贝的例子,自己改了改,欢迎大家品头论足。

  这段程序的输出为:

  objA1.RefClass.Field = 10
  objA2.RefClass.Field = 10
  objA1.RefClass.Field = 10
  objA2.RefClass.Field = 30
  objB1.RefClass.Field = 20
  objB2.RefClass.Field = 10  

  要注意的是,本例中实现深拷贝的ClassB类及其引用成员RefClass类必须添加Serializable特性。

 
 
using System;
using System.Collections;
using System.Collections.Generic;
using System.Text;
using System.IO;
using System.Runtime.Serialization.Formatters.Binary;

namespace ConsoleApplication1
{
// 被引用类,必须添加Serializable特性,否则不能实现序列化
[Serializable]
class RefClass
{
private int field;
public int Field
{
get { return field; }
set { this .field = value; }
}
}

// 浅拷贝示例
class ClassA : ICloneable
{
private RefClass refClass;
public RefClass RefClass
{
get { return refClass; }
set { refClass = value; }
}

public ClassA()
{
refClass
= new RefClass();
}

public object Clone()
{
// 调用MemberwiseClone实现浅拷贝
return MemberwiseClone();
}
}

// 深拷贝示例,必须添加Serializable特性,否则不能实现序列化
[Serializable]
class ClassB : ICloneable
{
private RefClass refClass;
public RefClass RefClass
{
get { return refClass; }
}

public ClassB()
{
refClass
= new RefClass();
}

// 深拷贝
public object Clone()
{
// 创建内存流
MemoryStream ms = new MemoryStream();
// 以二进制格式进行序列化
BinaryFormatter bf = new BinaryFormatter();
bf.Serialize(ms,
this );
// 反序列化当前实例到一个object
ms.Seek( 0 , 0 );
object obj = bf.Deserialize(ms);
// 关闭内存流
ms.Close();
return obj;
}
}

class Program
{
static void Main( string [] args)
{
// 浅拷贝
ClassA objA1 = new ClassA();
objA1.RefClass.Field
= 20 ;
ClassA objA2
= (ClassA)objA1.Clone();
// 在新对象中修改RefClass的Field值,源对象中的值亦更改
objA2.RefClass.Field = 10 ;
Console.WriteLine(
" objA1.RefClass.Field = " + objA1.RefClass.Field.ToString());
Console.WriteLine(
" objA2.RefClass.Field = " + objA2.RefClass.Field.ToString());
// 对新对象中的RefClass重新进行赋值操作,不影响源对象
objA2.RefClass = new RefClass();
objA2.RefClass.Field
= 30 ;
Console.WriteLine(
" objA1.RefClass.Field = " + objA1.RefClass.Field.ToString());
Console.WriteLine(
" objA2.RefClass.Field = " + objA2.RefClass.Field.ToString());
// 深拷贝
ClassB objB1 = new ClassB();
objB1.RefClass.Field
= 20 ;
ClassB objB2
= (ClassB)objB1.Clone();
// 在新对象中修改RefClass的Field值,源对象中的值不变
objB2.RefClass.Field = 10 ;
Console.WriteLine(
" objB1.RefClass.Field = " + objB1.RefClass.Field.ToString());
Console.WriteLine(
" objB2.RefClass.Field = " + objB2.RefClass.Field.ToString());
Console.ReadKey();
}
}
}
目录
相关文章
|
7月前
|
存储 C#
C#中的序列化和反序列化
C#中的序列化和反序列化
|
4月前
|
存储 C# 数据库
解决C#对Firebase数据序列化失败的难题
在游戏开发中,Unity结合Firebase实时数据库为开发者提供强大支持,但在C#中进行数据序列化和反序列化时常遇难题。文章剖析了数据丢失或反序列化失败的原因,并给出解决方案,包括使用`JsonUtility`、确保字段标记为`[Serializable]`以及正确配置网络请求。示例代码演示了如何在Unity环境中实现Firebase数据的序列化和反序列化,并通过设置代理IP、Cookies和User-Agent来增强网络请求的安全性。这些技巧有助于确保数据完整传输,提升开发效率。
解决C#对Firebase数据序列化失败的难题
|
7月前
|
XML 存储 JSON
c#XML、JSON的序列化和反序列化,看完你就懂了
c#XML、JSON的序列化和反序列化,看完你就懂了
133 0
|
7月前
|
存储 C#
C#中的序列化和反序列化案例
C#中的序列化和反序列化案例
|
7月前
|
XML 存储 JSON
C# 对象存储 (轻松实现序列化 | Xml | Json | 加密 | 压缩 | 注册表 | Redis)
开发时经常会遇到需要保存配置的情况,最常见的实现方式是将对象序列化成Json,再写入文件并保存到本地磁盘。 本文将使用开源库**ApeFree.DataStore**来替换原有的对象存储过程,实现一个可以随意切换存储方式的对象存储方法。 ApeFree.DataStore是一款可配置的对象存储库,支持在不同平台/介质中对内存中的对象进行存储与还原(如本地存储、注册表存储)。支持配置序列化格式(如Json、Xml),支持配置压缩算法(如GZip、Defalte),支持配置加密算法(如AES、RSA)。
136 0
C# 对象存储 (轻松实现序列化 | Xml | Json | 加密 | 压缩 | 注册表 | Redis)
|
7月前
|
XML 存储 JSON
C# | 使用Json序列化对象时忽略只读的属性
将对象序列化成为Json字符串是一个使用频率非常高的功能。Json格式具有很高的可读性,同时相较于XML更节省空间。 在开发过程中经常会遇到需要保存配置的场景,比如将配置信息保存在配置类型的实例中,再将这个对象序列化成为Json字符串并保存。当需要加载配置时,则是读取Json格式的字符串再将其还原成配置对象。在序列化的过程中,默认会将所有公开的属性和字段都序列化进入Json字符串中,这其中也会包含只读的属性或字段,而只读的属性和字段在反序列化的过程中其实是无意义的,也就是说这一部分存储是多余的。 本文将讲解如何在执行Json序列化时,忽略掉那些只读的属性和字段。
203 0
C# | 使用Json序列化对象时忽略只读的属性
|
JSON JavaScript 前端开发
c#JSON序列化&反序列化
JSON(全称为JavaScript ObjectNotation) 是一种轻量级的数据交换格式。它是基于JavaScript语法标准的一个子集。JSON采用完全独立于语言的文本格式,可以很容易在各种网络、平台和程序之间传输。JSON的语法很简单,易于人阅读和编写,同时也易于机器解析和生成。
79 0
|
XML 存储 C#
C#三十一 序列化与反序列化
C#三十一 序列化与反序列化
50 0
|
XML JSON JavaScript
30天C#基础巩固-----序列化,集合
30天C#基础巩固-----序列化,集合
123 0
30天C#基础巩固-----序列化,集合
|
JSON 开发框架 人工智能
Newtonsoft.Json C#Json序列化和反序列化工具的使用、类型方法大全
Newtonsoft.Json C#Json序列化和反序列化工具的使用、类型方法大全
695 0
Newtonsoft.Json C#Json序列化和反序列化工具的使用、类型方法大全