引用浅拷贝深拷贝

因为使用引用和拷贝的方法不对,尤其是在类的拷贝中,完全没有意识到这一点,结果花了很长时间来检查这个问题。最终发现这是浅拷贝和深拷贝的问题,而且这也是陈老师课上很重要的一部分,还是忘记了。。。!现在把这部分内容整理一下,而且在CSDN上看到一篇不错的文章,结合陈老师讲的,把这部分内容归纳一下。

 

概念:

  • 引用:用赋值运算符将对象a复制为对象b。此时a和b其实是同一个对象,只是名字不同而已,a的变化会立即影响到b,反之亦然。
  • 浅拷贝:用从object继承下来的MemberwiseClone方法,将对象a复制为对象b。此时,a和b的值类型字段是完全隔离的,一方的变化不会影响到另一方;a和b的引用类型字段则是同一个子对象,a中子对象的变化会立即影响到b中相应的子对象,反之亦然。string类型的子对象在浅拷贝中的行为与值类型子对象相同。
  • 深拷贝:用序列化和反序列化等方法将对象a复制为对象b,两个对象中的子对象完全隔离,一方的变化不会影响到另一方。深拷贝时不必知道对象内部有哪些数据

引用的例子:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.IO;


namespace Test
{
    public class MyData
    {
        static void Main(string[] args)
        {
            Person p1 = new Person(1, "Alan", new Car("宝马"));
            Person p2 = p1;//赋值仅是对引用的操作

            Console.Write("P1对象原始的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p1.id, p1.name, p1.car.name);
            Console.Write("P2对象原始的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p2.id, p2.name, p2.car.name);

            Console.WriteLine("\n改变P2的值后:");
            p2.id = 2;
            p2.name = "Charlie";
            p2.car.name = "红旗";

            Console.Write("P1对象现在的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p1.id, p1.name, p1.car.name);
            Console.Write("P2对象现在的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p2.id, p2.name, p2.car.name);
        }
    }

    class Car
    {
        public string name;
        public Car(string name)
        {
            this.name = name;
        }
    }

    class Person
    {
        public int id;
        public string name;
        public Car car;
        public Person(int id, string name, Car car)
        {
            this.id = id;
            this.name = name;
            this.car = car;
        }
    }
}


P1和P2其实是一个对象,输出结果:

浅拷贝的例子:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.IO;


namespace Test
{

    public class MyData
    {
        static void Main(string[] args)
        {
            Person p1 = new Person(1, "Alan", new Car("宝马"));
            Person p2 = p1.Clone() as Person;//浅拷贝调用

            Console.Write("P1对象原始的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p1.id, p1.name, p1.car.name);
            Console.Write("P2对象原始的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p2.id, p2.name, p2.car.name);

            Console.WriteLine("改变P1的值后:");
            p1.id = 2;
            p1.name = "Mary";
            p1.car.name = "红旗";

            //Console.WriteLine("改变P2的值后:");
            //p2.id = 2;
            //p2.name = "Mary";
            //p2.car.name = "红旗";

            Console.Write("P1对象现在的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p1.id, p1.name, p1.car.name);
            Console.Write("P2对象现在的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p2.id, p2.name, p2.car.name);
        }
    }

    class Car
    {
        public string name;
        public Car(string name)
        {
            this.name = name;
        }
    }

    class Person
    {
        public int id;
        public string name;
        public Car car;
        public Person(int id, string name, Car car)
        {
            this.id = id;
            this.name = name;
            this.car = car;
        }

        //对外提供一个创建自身的浅副本的能力
        public object Clone()
        {
            return this.MemberwiseClone();
        }
    }
}


值对象和string对象都是传值的,所以id和name都没有变化。而类是传递引用的,所以Car对象变化了。输出结果为:

深拷贝的例子:

参考的网站上的一个例子,使用了下面的代码:

    class Person : ICloneable //深拷贝必须实现ICloneable接口..   
    {
        public int id;
        public string name;
        public Car car;
        public Person(int id, string name, Car car)
        {
            this.id = id;
            this.name = name;
            this.car = car;
        }

        //对外提供一个创建自身的浅副本的能力
        public object Clone()
        {
            return this.MemberwiseClone() as Person;
        }
    }

发现完全不能实现,跟浅拷贝的作用是一样的!又查阅了其它资料,只要用MemberwiseClone()就只是浅拷贝而已。所以深拷贝要么一个一个的属性重新赋值,最简单最粗暴的方法。但是肯定是扩展性不好,所以采纳陈老师课上的另外一种方法,采用序列化和反序列化的办法,这样不用去关心这个类到底有什么属性了。

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


namespace Test
{

    public class MyData
    {
        static void Main(string[] args)
        {
            Person p1 = new Person(1, "Alan", new Car("宝马"));
            Person p2 = (Person)p1.DeepCopy(p1);//深拷贝调用

            Console.Write("P1对象原始的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p1.id, p1.name, p1.car.name);
            Console.Write("P2对象原始的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p2.id, p2.name, p2.car.name);

            Console.WriteLine("改变P1的值后:");
            p1.id = 2;
            p1.name = "Mary";
            p1.car.name = "红旗";

            //Console.WriteLine("改变P2的值后:");
            //p2.id = 2;
            //p2.name = "Mary";
            //p2.car.name = "红旗";

            Console.Write("P1对象现在的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p1.id, p1.name, p1.car.name);
            Console.Write("P2对象现在的值:");
            Console.WriteLine("id:{0}  name:{1}  car:{2}", p2.id, p2.name, p2.car.name);
        }
    }
    [Serializable]
    class Car
    {
        public string name;
        public Car(string name)
        {
            this.name = name;
        }
    }
    [Serializable]
    class Person //深拷贝必须实现ICloneable接口..   
    {
        public int id;
        public string name;
        public Car car;
        public Person(int id, string name, Car car)
        {
            this.id = id;
            this.name = name;
            this.car = car;
        }

        //对外提供一个创建自身的浅副本的能力
        public object Clone()
        {
            return this.MemberwiseClone() as Person;
        }

        //深拷贝
        public object DeepCopy(object obj)
        {
            MemoryStream ms = new MemoryStream();
            BinaryFormatter bf = new BinaryFormatter();
            bf.Serialize(ms, obj);
            ms.Position = 0;
            return bf.Deserialize(ms);
        }
    }
}


注意[Serializable]以及添加的命名空间。输出的结果为:


字符串的解释:

字符串在C#中其实中引用类型,但是使用浅拷贝时,发现象值类型一样处理了。这是为什么?来看MSDN的解释:字符串对象是“不可变的”,即它们一旦创建就无法更改。对字符串进行操作的方法实际上返回的是新的字符串对象。

比如下面这个例子:

string s1 = "hello";
string s2 = s1;
s1 = "world";
Console.WriteLine("s1:{0}/ns2:{1}", s1, s2);

我们平常大量使用这样的方法,没有象引用那样造成麻烦,string s2=s1;语句将会使s1 ,s2指向同一内存位置.如果改变s1(s1="world"),原来的"hello"并不会改变,只是会创建新的字符串对象"world",s1对其引用。


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

极地语音工作室

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值