C#浅拷贝和深拷贝实例解析

 更新时间:2014年08月13日 10:52:43   投稿:shichen2014  
这篇文章主要介绍了C#浅拷贝和深拷贝,是比较重要的概念,需要的朋友可以参考下

在有些时候,我们需要从数据库读取数据填充对象或从硬盘读取文件填充对象,但是这样做相对耗时。这时候我们就想到了对象的拷贝。本文即以实例形式解析了C#浅拷贝和深拷贝的用法。具体如下:

一、浅拷贝

1.什么是"浅拷贝":

当针对一个对象前拷贝的时候,对于对象的值类型成员,会复制其本身,对于对象的引用类型成员,仅仅复制对象引用,这个引用指向托管堆上的对象实例。

2.有一个对象,包含引用类型的类成员和值类型的struct成员

Cinema包含引用类型成员Room和值类型成员Film。

  public class Room
  {
    public int _maxSeat;
 
    public Room(int maxSeat)
    {
      this._maxSeat = maxSeat;
    }
  }
 
  public struct Film
  {
    public string _name;
 
    public Film(string name)
    {
      this._name = name;
    }
  }
 
  public class Cinema
  {
    public Room _room;
    public Film _film;
 
    public Cinema(Room room, Film film)
    {
      this._room = room;
      this._film = film;
    }
 
    public object Clone()
    {
      return MemberwiseClone(); //对引用类型实施浅复制
    }
  }
 

3.测试拷贝后的效果

①打印出原先对象拷贝前值类型和引用类型成员的值
②对原先对象拷贝,打印出复制对象值类型和引用类型成员的值
③改变原先对象的值,再次打印原先对象的值类型和引用类型成员的值
④再次打印复制对象值类型和引用类型成员的值

static void Main(string[] args)
{
  Room room1 = new Room(60);
  Film film1 = new Film("家园防线");
  Cinema cinema1 = new Cinema(room1, film1);
  Cinema cinema2 = (Cinema)cinema1.Clone();
  Console.WriteLine("拷贝之前,结构成员的字段值为{0},引用类型成员的字段值为{1}", cinema1._film._name,cinema1._room._maxSeat);

  Console.WriteLine("拷贝之后,新的结构成员的字段值为{0},引用类型成员的字段值为{1}", cinema2._film._name, cinema2._room._maxSeat);

  //修改拷贝之前引用类型的字段值
  cinema1._film._name = "极品飞车";
  cinema1._room._maxSeat = 80;

  Console.WriteLine("修改之后,结构成员的字段值为{0},引用类型成员的字段值为{1}", cinema1._film._name, cinema1._room._maxSeat);
  Console.WriteLine("修改之后,新的结构成员的字段值为{0},引用类型成员的字段值为{1}", cinema2._film._name, cinema2._room._maxSeat);

  Console.ReadKey();
}

运行结果如下:

分析:

浅拷贝关键点是对引用类型拷贝的是对象引用,这个引用指向托管堆上的对象实例。改变原对应引用类型的值,会影响到复制对象。

二、深拷贝

1.什么是"深拷贝"

对引用成员指向的对象也进行复制,在托管堆上赋值原先对象实例所包含的数据,再在托管堆上创建新的对象实例。

2.通过对每个对象成员进行复制进行深拷贝

    public object Clone()
    {
      Room room = new Room();
      room._maxSeat = this._room._maxSeat;//复制当前引用类型成员的值到新对象 
      Film film = this._film; //值类型直接赋值
      Cinema cinema = new Cinema(room, film);
      return cinema;
    }
 

3.也可以通过序列化和反序列化进行深拷贝

    public object Clone1()
    {
      BinaryFormatter bf = new BinaryFormatter();
      MemoryStream ms = new MemoryStream();
      bf.Serialize(ms, this); //复制到流中
      ms.Position = 0;
      return (bf.Deserialize(ms));
    }

4.采用序列化和反序列化深拷贝,但必须把所有的类打上[Serializable],测试代码如下:

  [Serializable]
  public class Room
  {
    public int _maxSeat;
 
    public Room()
    {}
 
    public Room(int maxSeat)
    {
      this._maxSeat = maxSeat;
    }
  }
 
  [Serializable]
  public struct Film
  {
    public string _name;
 
    public Film(string name)
    {
      this._name = name;
    }
  }
 
  [Serializable]
  public class Cinema
  {
    public Room _room;
    public Film _film;
 
    public Cinema(Room room, Film film)
    {
      this._room = room;
      this._film = film;
    }
 
    //浅拷贝
    //public object Clone()
    //{
    //  return MemberwiseClone(); //对引用类型实施浅复制
    //}
 
    //深拷贝 对每个对象成员进行复制
    public object Clone()
    {
      Room room = new Room();
      room._maxSeat = this._room._maxSeat;//复制当前引用类型成员的值到新对象 
      Film film = this._film; //值类型直接赋值
      Cinema cinema = new Cinema(room, film);
      return cinema;
    }
 
    //使用序列化和反序列化进行复制
    public object Clone1()
    {
      BinaryFormatter bf = new BinaryFormatter();
      MemoryStream ms = new MemoryStream();
      bf.Serialize(ms, this); //复制到流中
      ms.Position = 0;
      return (bf.Deserialize(ms));
    }
  }

5.测试拷贝后的效果

①打印出原先对象拷贝前值类型和引用类型成员的值
②对原先对象拷贝,打印出复制对象值类型和引用类型成员的值
③改变原先对象的值,再次打印原先对象的值类型和引用类型成员的值
④再次打印复制对象值类型和引用类型成员的值

 static void Main(string[] args)
    {
      Room room1 = new Room(60);
      Film film1 = new Film("家园防线");
      Cinema cinema1 = new Cinema(room1, film1);
      Cinema cinema2 = (Cinema)cinema1.Clone1();
      Console.WriteLine("拷贝之前,结构成员的字段值为{0},引用类型成员的字段值为{1}", cinema1._film._name,cinema1._room._maxSeat);
 
      Console.WriteLine("拷贝之后,新的结构成员的字段值为{0},引用类型成员的字段值为{1}", cinema2._film._name, cinema2._room._maxSeat);
 
      //修改拷贝之前引用类型的字段值
      cinema1._film._name = "极品飞车";
      cinema1._room._maxSeat = 80;
 
      Console.WriteLine("修改之后,结构成员的字段值为{0},引用类型成员的字段值为{1}", cinema1._film._name, cinema1._room._maxSeat);
      Console.WriteLine("修改之后,新的结构成员的字段值为{0},引用类型成员的字段值为{1}", cinema2._film._name, cinema2._room._maxSeat);
 
      Console.ReadKey();
    }

结果:

 

分析:

深拷贝后,两个对象的引用成员已经分离,改变原先对象引用类型成员的值并不会对复制对象的引用类型成员值造成影响。

相关文章

  • VS2015 C#生成dll文件的方法(32/64)

    VS2015 C#生成dll文件的方法(32/64)

    这篇文章主要介绍了VS2015 C#生成dll文件的方法(32/64),需要的朋友可以参考下
    2016-12-12
  • C# 实现WebSocket服务端教程

    C# 实现WebSocket服务端教程

    这篇文章主要介绍了C# 实现WebSocket服务端教程,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-10-10
  • C#实现表格数据转实体的示例代码

    C#实现表格数据转实体的示例代码

    在实际开发过程中,特别是接口对接之类的,对于这种需求是屡见不鲜,现在很多在线平台也都提供了像json转实体、sql转实体等。本文将用C#实现这一功能,需要的可以参考一下
    2022-09-09
  • 关于c#中枚举类型支持显示中文的扩展说明

    关于c#中枚举类型支持显示中文的扩展说明

    需求 : 枚举类型在界面显示的时候可以显示相应的中文信息, 这样界面对用户友好 . 场景 : 在一些业务中涉及到审核功能的时候, 往往有这几个状态 :未送审 , 审核中 ,审核通过, 驳回 . 这个时候我们会定义一个枚举类型来描述 :
    2013-03-03
  • C# wpf使用GDI+实现截屏效果

    C# wpf使用GDI+实现截屏效果

    wpf做屏幕录制或者屏幕广播之类的功能时需要实现截屏,在C#中比较容易实现的截屏方法是使用GDI+,所以本文将展示一下如何使用GDI+实现截屏,需要的可以参考下
    2023-09-09
  • C#实现解析百度天气数据,Rss解析百度新闻以及根据IP获取所在城市的方法

    C#实现解析百度天气数据,Rss解析百度新闻以及根据IP获取所在城市的方法

    这篇文章主要介绍了C#实现解析百度天气数据,Rss解析百度新闻以及根据IP获取所在城市的方法,非常具有实用价值,需要的朋友可以参考下
    2014-10-10
  • C#绘制鼠标指针的示例代码

    C#绘制鼠标指针的示例代码

    这篇文章主要为大家详细介绍了如何利用C#实现将鼠标的指针样式给绘制成图片,显示或者保存下来,文中的示例代码讲解详细,需要的可以参考一下
    2024-01-01
  • C#中的那些警告该如何去除(完全去除C#警告)

    C#中的那些警告该如何去除(完全去除C#警告)

    C#(英文名为 CSharp)是微软开发的一种面向对象的编程语言,下面这篇文章主要给大家介绍了关于C#中的那些警告该如何去除的相关资料,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2023-02-02
  • C#实现Excel合并单元格数据导入数据集详解

    C#实现Excel合并单元格数据导入数据集详解

    这篇文章主要为大家详细介绍了C#如何实现Excel合并单元格数据导入数据集,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-01-01
  • C# 站点IP访问频率限制 针对单个站点的实现方法

    C# 站点IP访问频率限制 针对单个站点的实现方法

    下面小编就为大家带来一篇C# 站点IP访问频率限制 针对单个站点的实现方法。小编觉的挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-12-12

最新评论