【转】传递引用类型参数(C#)
时间:2011-03-29 来源:冀中
MSDN:
引用类型的变量不直接包含其数据;它包含的是对其数据的引用。当通过值传递引用类型的参数时,有可能更改引用所指向的数据,如某类成员的值。但是无法更改引用本身的值;也就是说,不能使用相同的引用为新类分配内存并使之在块外保持。若要这样做,应使用 ref 或 out 关键字传递参数。为了简单起见,下面的示例使用 ref。
下面的示例演示通过值向 Change 方法传递引用类型的参数 arr。由于该参数是对 arr 的引用,所以有可能更改数组元素的值。但是,试图将参数重新分配到不同的内存位置时,该操作仅在方法内有效,并不影响原始变量 arr。
class PassingRefByVal
{
static void Change(int[] pArray)
{
pArray[0] = 888; // This change affects the original element.
pArray = new int[5] {-3, -1, -2, -3, -4}; // This change is local.
System.Console.WriteLine("Inside the method, the first element is: {0}", pArray[0]);
}
static void Main()
{
int[] arr = {1, 4, 5};
System.Console.WriteLine("Inside Main, before calling the method, the first element is: {0}", arr [0]);
Change(arr);
System.Console.WriteLine("Inside Main, after calling the method, the first element is: {0}", arr [0]);
}
}
输出 Inside Main, before calling the method, the first element is: 1
Inside the method, the first element is: -3
Inside Main, after calling the method, the first element is: 888
代码讨论 在上个示例中,数组 arr 为引用类型,在未使用 ref 参数的情况下传递给方法。在此情况下,将向方法传递指向 arr 的引用的一个副本。输出显示方法有可能更改数组元素的内容,在这种情况下,从 1 改为 888。但是,在 Change 方法内使用 new 运算符来分配新的内存部分,将使变量 pArray 引用新的数组。因此,这之后的任何更改都不会影响原始数组 arr(它是在 Main 内创建的)。实际上,本示例中创建了两个数组,一个在 Main 内,一个在 Change 方法内。 以上引用地址:http://msdn.microsoft.com/zh-cn/library/s6938f28(v=VS.80).aspx
另,示例:
protected void Button1_Click(object sender, EventArgs e)
{
Class1 obj1 = new Class1();
obj1.Name = "name1";
obj1.ID = "1";
Class1 obj2 = GetObj(obj1);
Response.Write("obj1 => ID:"+obj1.ID+";Name:"+obj1.Name);
Response.Write("<br/>");
Response.Write("obj2 => ID:" + obj2.ID + ";Name:" + obj2.Name);
}
protected Class1 GetObj(Class1 c)
{
c.ID = "3";
Class1 b = new Class1();
b.ID = "2";
b.Name = "name2";
c = b;
return c;
}
输出结果:
obj1 => ID:3;Name:name1
obj2 => ID:2;Name:name2
个人理解:
对象作为参数传递时,传递的是对象的引用(即:使用地址)。比如原来对象(obj1)在地址22处,那么在GetObj方法内,首先对c进行修改,那么修改的也是地址22处的对象,即obj1对象。当执行语句c=b;时,那么原本指向22地址的指针被赋予了新的地址,比如33,那么再对c进行各种操作,那么操作的目标也只能是地址33处的新对象,与地址为22处的对象obj1再也不发生任何关系了。因此也就出现了以上结果。
由此扩展并推断出:在方法参数传递时,不加out、ref的情况下,c#都是按值传递参数的。对于值类型的参数,拷贝的是值本身。对于引用型对象,也进行拷贝,只不过是拷贝了该对象的指针!
在加out、ref情况下,一旦方法内部有new之类内存重新分配之类的语句,系统将对原对象进行重新分配,即将原对象指针将指向新对象地址!
理解不当之处,敬请指出。