• 08浅拷贝和深拷贝


    有时候,从数据库读取数据填充对象或从硬盘读取文件填充对象,这样相对耗时。这时候想到了对象的拷贝。

      浅拷贝

    □ 什么是"浅拷贝"

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

     

    □ 有一个对象,包含引用类型的类成员和值类型的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(); //对引用类型实施浅复制
            }
        }
     

    □ 测试拷贝后的效果

    1、打印出原先对象拷贝前值类型和引用类型成员的值
    2、对原先对象拷贝,打印出复制对象值类型和引用类型成员的值
    3、改变原先对象的值,再次打印原先对象的值类型和引用类型成员的值
    4、再次打印复制对象值类型和引用类型成员的值

            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

     

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

     

      深拷贝

    □ 什么是"深拷贝"

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

     

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

            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));
            }

     

    □ 测试,采用序列化和反序列化深拷贝,但必须把所有的类打上[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));
            }
        }
     

    □ 测试拷贝后的效果

    1、打印出原先对象拷贝前值类型和引用类型成员的值
    2、对原先对象拷贝,打印出复制对象值类型和引用类型成员的值
    3、改变原先对象的值,再次打印原先对象的值类型和引用类型成员的值
    4、再次打印复制对象值类型和引用类型成员的值

     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();
            }
     

    结果:
    2

     

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

     

    参考资料
    《.NET之美》--张子阳,感谢写了这么好的书!

  • 相关阅读:
    拼接个URL你也能搞错,还写个屁的爬虫
    在 Python 中接管键盘中断信号
    关于re.sub从匹配的文本中处理后之际替换匹配匹配到的数据
    loguru 如何把不同的日志写入不同的文件中
    Linux 中,没有 zip命令,怎么生成 zip 文件?
    mysql查询tinyint变成true,false;
    muislider设置选项卡,muislider设置默认选项卡
    js数组排序 js排序
    "enablePullDownRefresh": false,mui配置是否下拉刷新,mui配置下拉刷新
    Webview窗口刷新
  • 原文地址:https://www.cnblogs.com/darrenji/p/3601523.html
Copyright © 2020-2023  润新知