请看下面代码:
static void Main(string[] args)
{
int[] a = new int[]{ 1, 2, 3 };
Test_1(a);
Console.WriteLine(string.Join(",", a));
Test_2(a);
Console.WriteLine(string.Join(",", a));
Console.ReadLine();
}
private static void Test_1(int[] a)
{
a = new int[] { 100, 200, 300 };
}
private static void Test_2(int[] a)
{
a[1] = 10;
}
请看输出结果:
是不是觉得很意外?!
为了更好地解释清楚,我将上面的两个 Test 方法的参数名改一下:
private static void Test_1(int[] b)
{
b = new int[] { 100, 200, 300 };
}
private static void Test_2(int[] c)
{
c[1] = 10;
}
首先需要说明的是,数组是引用类型,参数传递的时候是引用传递(传递内存地址)的。
当调用Test_1方法时,CLR在栈上创建了一个变量b,并将变量a保存的堆地址赋值给变量b(变量a与变量b是两个独立的变量,虽然它们都指向了同一个堆地址,理解这一点很重要)。
Test_1方法内部 new 了一个新的数组,CLR会开辟一个新的堆空间,并将该堆地址赋值给变量b,此时变量a保存的堆地址不变,也就是说此时变量a与变量b分别指向了不同的堆地址,所以变量a的值并没有变。
下面上图,能更好的理解(画的不好不要介意哈):
至于Test_2方法,变量c与变量a都是指向同一个堆地址,所以改变了变量c的值也就导致变量a的值变了。
参考链接: