乐趣区

C中对引用类型的误解

C# 中对引用类型的误解

值类型和引用类型作为两个非常基础而且很重要的概念,一般我们都是在最开始的时候学的,你听到的可能是这样的:值类型传递的是具体的值(副本),引用类型传递的是引用,对于前者大家都不会有什么疑惑,但是在引用类型上就可能会进入一些误区。

概念插入:

值类型:值类型传递的是数据的副本,也就是将整个数据进行 Copy 然后赋值给另一个变量。

引用类型:传递的是对象的地址 / 路径(一般都会叫做引用),在对象的赋值过程中实际上是把目标对象的地址以副本的形式给了接收的对象。

实例说明

如下我们新建了一个 Student 类,定义了一个变量 student1,然后定义了一个变量 student2 并将 student1 赋值给了它,紧接着创建了 student3 并将 student3 赋值给了 student1。

输出结果:1

class Program
{static void Main(string[] args)
    {
        //step1
        Student student1 = new Student() { StudentId = 1};
        Student student2 = student1;
        Student student3 = new Student() { StudentId = 3};
        //step2
        student1 = student3;
        Console.WriteLine(student2.StudentId.ToString());
        Console.ReadKey();}
}
public class Student
{public int StudentId { get; set;}
}

Step1 的操作如下图所示:

在图中的右侧我们使用矩形代替两个通过 new 关键字创建的对象,并为其标识了两个虚拟的 ID,在左侧我们定义了三个变量,如图的圆角矩形所示,这三个变量不是真正意义上的对象,只是对象的一个所在地址而已,也就是我们通过这个变量就可以找到一个特定的地址。其中 student1 变量持有的 ID(内存地址)为 1001,student2 也是 1001,student3 为 1002。这里需要注意的是 student1 的 1001 和 student2 的 1001 虽然都是指向的 1001,但是这两个 1001 都只是对象地址的一个副本,也就是说 student1 的 1001 和 student2 的 1001 并不是同一个,只是被 Copy 了而已(可以理解为这个地址是以值类型的方式进行传递的)。

这里要注意的是变量 student2 和 student1 是没有关系的,它们只和右侧的 1001 有关系,只是它们地址的值是相等的。

new 关键字:

创建特定对象的实例并返回其引用地址(只是返回了地址的一个副本,而不是真正的对象)

When the object is created, the memory is allocated on the managed heap, and the variable holds only a reference to the location of the object.(来源于官方文档:https://docs.microsoft.com/en…)

译:创建对象后,内存会在托管堆上进行分配,并且变量只保留对对象位置的引用。

Step2 的操作如下图所示:

在 Step2 的代码中我们将 student3 的变量赋值给了 student1,那么这段代码执行的是将 student3 对应的 ID(引用地址)以副本的方式给了 student1。

//step2
student1 = student3;
Console.WriteLine(student2.StudentId.ToString());
Console.ReadKey();

如果你对引用传递理解有有偏差的话你可能会认为 student2 的地址也会变成 1002,也就是 student2.StudentId 变成了 3。这样理解是不正确的,我们刚才说过了虽然它们的引用地址是都是 1001,但是这个地址也是通过副本进行传递的。也就是说当 student1 的地址发生改变并不会影响 student2 的地址。

再述:

当右侧对象中的值发生改变时,通过左侧变量来获取拿到的就是改变后的值,而当左侧变量值的改变时并不会影响右侧对象值的。左侧变量值的改变只是对应的地址发生了改变而不是对象的值发生了改变。

举例分析

这里从我们常用的两个购物平台淘宝、京东来举例说明引用地址传递。

右侧有两个对象分别是淘宝和京东,左侧我们定义了三个变量来保存这个对象的地址(和上边的例子是一样的)

如下图,我们将 valA 变量存放的地址改成了 www.jd.com,我们只是改变了变量存放的地址而并没有改变 valA 之前所对应的对象的值。

总结

引用类型传递的是对象的引用,是将引用地址以副本的方式进行传递。

退出移动版