日期:2014-05-18  浏览次数:20956 次

ref和out的区别?
ref和out有什么区别?

------解决方案--------------------
ref 是按地址传值
out ,,
------解决方案--------------------
在C# 中,既可以通过值也可以通过引用传递参数。通过引用传递参数允许函数成员更改参数的值,并保持该更改。若要通过引用传递参数, 可使用ref或out关键字。ref和out这两个关键字都能够提供相似的功效,其作用也很像C中的指针变量。它们的区别是:

1、使用ref型参数时,传入的参数必须先被初始化。对out而言,必须在方法中对其完成初始化。 

2、使用ref和out时,在方法的参数和执行方法时,都要加Ref或Out关键字。以满足匹配。 

3、out适合用在需要retrun多个返回值的地方,而ref则用在需要被调用的方法修改调用者的引用的时候。 

注:在C#中,方法的参数传递有四种类型:传值(by value),传址(by reference),输出参数(by output),数组参数(by array)。传值参数无需额外的修饰符,传址参数需要修饰符ref,输出参数需要修饰符out,数组参数需要修饰符params。传值参数在方法调用过程中如果改变了参数的值,那么传入方法的参数在方法调用完成以后并不因此而改变,而是保留原来传入时的值。传址参数恰恰相反,如果方法调用过程改变了参数的值,那么传入方法的参数在调用完成以后也随之改变。实际上从名称上我们可以清楚地看出两者的含义--传值参数传递的是调用参数的一份拷贝,而传址参数传递的是调用参数的内存地址,该参数在方法内外指向的是同一个存储位置。

方法参数上的 ref 方法参数关键字使方法引用传递到方法的同一个变量。当控制传递回调用方法时,在方法中对参数所做的任何更改都将反映在该变量中。

若要使用 ref 参数,必须将参数作为 ref 参数显式传递到方法。ref 参数的值被传递到 ref 参数。

传递到 ref 参数的参数必须最先初始化。将此方法与 out 参数相比,后者的参数在传递到 out 参数之前不必显式初始化。

属性不是变量,不能作为 ref 参数传递。

如果两种方法的声明仅在它们对 ref 的使用方面不同,则将出现重载。但是,无法定义仅在 ref 和 out 方面不同的重载。

out

方法参数上的 out 方法参数关键字使方法引用传递到方法的同一个变量。当控制传递回调用方法时,在方法中对参数所做的任何更改都将反映在该变量中。

当希望方法返回多个值时,声明 out 方法非常有用。使用 out 参数的方法仍然可以返回一个值。一个方法可以有一个以上的 out 参数。

若要使用 out 参数,必须将参数作为 out 参数显式传递到方法。out 参数的值不会传递到 out 参数。

不必初始化作为 out 参数传递的变量。然而,必须在方法返回之前为 out 参数赋值。

属性不是变量,不能作为 out 参数传递。

 


------解决方案--------------------
ref ,out 其实都可以说是引用传递,区别于值传递。
引用传递 的时候 调用函数对参数的修改会反映到原来的传入的参数上,
但值传递则不会改变。传递的只是原来变量的 副本。

ref : 传递前必须初始化,
out:不必初始化,就算初始化,也会被无视,
out 作为输出参数,可以解决函数只有一个返回值的问题,


------解决方案--------------------
也就是说ref类似于c语言中的指针。

 using System;

class Account
{
 private int balance = 0; //字段
 public int Balance //属性
 {
get { return balance; }
set { balance = value;}
 }
 public void Deposit(int n)
 { this.balance += n; }

 public void WithDraw(int n)
 { this.balance -= n; }
}

class Client
{
 public static void Main()
 {
Account a = new Account();
a.Balance = 1000; // 可以读写属性,因为属性Balance是public型的
//a.balance = 1000; //不可以读写字段,因为字段balance是private型的

a.WithDraw(500);
a.Deposit(2000);
Console.WriteLine("before Method call: a.balance = {0}", a.Balance);
  
int x = 1;
int y = 100;
int z;
AMethod(x, ref a.balance ,out z); //ref a.balance 正确,a.balance是字段变量
//AMethod(x, ref a.Balance ,out z); //ref a.Balance 错误,a.Balance是属性
y = a.balance;
Console.WriteLine("After Method Call : x = {0}, y = {1}, z = {2}", x, y, a.balance);
AMethod(x, ref y ,out a.balance); //out a.balance 正确,a.balance是字段变量
//AMethod(x, ref y ,out a.Balance); //ref a.Balance 错误,a.Balance是字段变量
z = a.balance;
Console.WriteLine("After Method Call : x = {0}, y = {1}, z = {2}", x, y, a.balance);
AMethod(x, ref y ,out z);
Console.WriteLine("After Method Call : x = {0}, y = {1}, z = {2}", x, y, z);
 }

 public static void AMethod(int x, ref int y, out int z)
 {
x = 7;
y = 8;
z = 9;
 }
}

输出结果:

---------- C# Run ----------
before Method call: a.balance = 2500
After Method Call : x = 1, y = 8, z = 8
After Method Call : x = 1, y = 8, z = 9
After Method Call : x = 1, y = 8, z = 9

输出完成 (耗时: 0 秒) - 正常终止
------解决方案--------------------
ref和out都是按引用传递,不同的是ref在传递前需要显式初始化