c#引用参数传递的深入剖析
值类型的变量存储数据,而引用类型的变量存储对实际数据的引用。(这一点很重要,明白了之后就能区分开值类型和引用类型的差别)
在参数传递时,值类型是以值的形式传递的(传递的是值,对变量本身没有影响),是将要传递的参数的值复制给函数的形参,因此在函数体类对于该形参的任何改变都不会影响原来的值;
引用类型是以对象引用的形式传递的(传递的是引用,也就是说把同一个父级传过去,拥有相同的父亲),是将要传递的对象的引用复制给函数的形参,这时形参是实参引用的复制,注意:是引用的复制,而不是原引用,和原引用指向相同的对象,因此对于引用对象所做的更改将会直接影响原来的值,但是对于引用本身,在函数内的任何改变将不会影响原引用。
先列出前辈出的两个例子:
1 class A { 2 public string data=""; 3 } 4 class Program { 5 static void F( A a1) { 6 //a1指向传来的对象a http://www.cnblogs.com/roucheng/ 7 a1.data = "2";//修改a1指向的对象 8 a1 = new A();//a1指向另一个对象,注意,这时a1已经不指向原来的对象a了,而原来的引用还是指向对象a 9 a1.data = "3";//修改新建的对象,不会影响原来对象a的值 10 } 11 static void Main() { 12 A a = new A();//实例化A的一个对象,并用a1指向该对象 a.data = "1";//将a的data字段赋值为"1" 13 F(a);//调用函数F,注意:这时将对象a的引用(不是对象a)赋值给参数a1, 14 Console.WriteLine(a.data); } }
这是一个直接传递的例子,在F(a)调用时,是把值传递过去,a是引用参数,所以传递的是引用,也就是说把a的引用当做值传递给了F()函数中的a1,在F()函数中对a1操作:a1.data = "2";就改变了a的值,当执行a1 = new A();时,a1的引用初始化,也就是说不再是传过来的那个,所以在执行a1.data = "3";是对a没有影响。下面一个例子你会看到传递的不是值而就是引用,也就是说把引用传给了a1,在a1 = new A();时,改变的是引用,也就是说引用的父亲本身就变了 所以也就是改变了a的引用,最后a的值变成了3
ref 串参数:ref 关键字使参数按引用传递。其效果是,当控制权传递回调用方法时,在方法中对参数所做的任何更改都将反映在该变量中。若要使用 ref 参数,则方法定义和调用方法都必须显式使用 ref 关键字。例如:对于值类型,可以向上面的引用串参数一样传递,对于已经是引用类型的参数,大家可能会说那不是多此一举吗?其实不然,因为其中的实机理完全不一样:考查上个示例的变种
1 class A { 2 public string data=""; 3 } 4 class Program { 5 static void F( ref A a1) { 6 //a1和a是同一个实例,而不是指向同一对象的引用,即a1和a在存在于内存中的地址是一样的 7 a1.data = "2";//修改a1指向的对象 8 a1 = new A();//a1指向另一个对象,理所当然别名a也指向该对象了,注意,这时原来的对象已经没有任何引用指向了,因此,可以说原来的对象已经不可访问了。 9 a1.data = "3";//修改新建的对象的属性 } static void Main() { 10 A a = new A();//实例化A的一个对象,并用a1指向该对象 a.data = "1";//将a的data字段赋值为"1" 11 F(ref a);//调用函数F,注意:这时将对象a的引用传给a1,不是赋值,相当与 给a对象的引用起了个别名 12 Console.WriteLine(a.data);//这时a已经指向函数中新建的对象,因此值应为"3" 13 }
可以这么理解,没有ref时的引用对象的参数传递就相当于c++中的一般指针传递(函数声明相当于: void F(Type * v)),而有ref时的引用对象的参数传递相当于c++中的一般指向指针的指针传递(函数声明相当于: void F(Type ** v)).