你想要返回多个结果吗?来看看吧 :Ref和Out关键字异同

类型介绍

在几乎所有的OOP语言中,都存在2种类型的值。

  1. 值类型
  2. 引用类型

以C#为例:其值类型为sbyte,byte,char,short,ushort,int,uint,long和ulong,float和double,当然还有decimal和bool。而引用类型则是string和object。

我想说的

我想说的就是——Ref和Out把我弄糊涂的原因是,当时没有认真的去分析它对不同类型所做出的不同的动作。

对于值类型。

使用了Ref和Out的效果就几乎和C中使用了指针变量一样。它能够让你直接对原数进行操作,而不是对那个原数的Copy进行操作。举个小例子:

using System;namespace ConsoleApplication4
{
/// <summary>
/// Class1 的摘要说明。
/// </summary>
class Class1
{
/// <summary>
/// 应用程序的主入口点。
/// </summary>
[STAThread]
static void Main(string[] args)
{
int a = 5;
int b; squareRef(ref a);
squareOut(
out b); Console.WriteLine("The a in the Main is: " + a);
Console.WriteLine(
"The b in the Main is: " + b);
}
static void squareRef(ref int x)
{
x
= x * x;
Console.WriteLine(
"The x in the squareRef is: " + x);
}
static void squareOut(out int y)
{
y
= 10;
y
= y * y;
Console.WriteLine(
"The y in the squareOut is: " + y);
}
}
}

  

显示的结果就是——25 100 25 100。

这样的话,就达到了和C中的指针变量一样的作用。

对于引用类型。

对于引用类型就比较难理解了。

先要了解到这一层——就是当一个方法接收到一个引用类型的变量的时候,它将获得这个引用(Reference)的一个Copy。由于Ref关键字可以用来向方法传递引用。所以,如果这个功能被误用了——比如:当一个如数组类型的引用对象用关键字Ref传递的时候,被调用的方法实际上已经控制了传递过来的引用本身。这样将使得被调用方法能用不同的对象甚至NULL来代替调用者的原始引用!

如图。内存地址为2000的变量arrayA中其实存放着数组{1,2,3,4,……}的内存起始地址10000。如果一个方法fun()使用fun( arrayA[] )的话,它将顺利的获得数据10000,但这个10000将放在一个Copy中,不会放到内存的2000位置。而这个时候我们如果使用fun( ref arrayA[] )的话,我们得到的值就是2000啦(也就是说,被调用方法能够修改掉arrayA中的那个引用,使之不再指向10000,甚至可以用NULL来代替10000,这样的话,那个10000地址中的数据可能就要被垃圾回收机制清理了。)

01 using System;
02  class TestApp
03  {
04  static void outTest(out int x, out int y)
05  {//离开这个函数前,必须对x和y赋值,否则会报错。
06  //y = x;
07  //上面这行会报错,因为使用了out后,x和y都清空了,需要重新赋值,即使调用函数前赋过值也不行
08  x = 1;
09  y = 2;
10  }
11  static void refTest(ref int x, ref int y)
12  {
13  x = 1;
14  y = x;
15  }
16  public static void Main()
17  {
18  //out test
19  int a,b;
20  //out使用前,变量可以不赋值
21  outTest(out a, out b);
22  Console.WriteLine("a={0};b={1}",a,b);
23  int c=11,d=22;
24  outTest(out c, out d);
25  Console.WriteLine("c={0};d={1}",c,d);
26  //ref test
27  int m,n;
28  //refTest(ref m, ref n);
29  //上面这行会出错,ref使用前,变量必须赋值
30  int o=11,p=22;
31  refTest(ref o, ref p);
32  Console.WriteLine("o={0};p={1}",o,p);
33  }
34  }

运行结果如下:

下法我解析一下为什么是这样的结果:

ref是传递参数的地址,out是返回值,两者有一定的相同之处,不过也有不同点。

  - 使用ref前必须对变量赋值,out不用。

  - out的函数会清空变量,即使变量已经赋值也不行,退出函数时所有out引用的变量都要赋值,ref引用的可以修改,也可以不修改。

在上面的代码中:

首先看 outTest(out a, out b)的输出结果

结果是 a=1;b=2;这是在 static void outTest(out int x, out int y)函数定义时的x,y的值

x=1;y=2;然后就直接传递给a,b了

接下来再看 outTest(out c, out d);

这个与 outTest(out a, out b)的不同之处在于

他前面己给 c,d赋值

int c = 11, d = 22;

一般常理来说,这会outTest(out c, out d)的输出值应为 c=11;d=22;

但是真实结果却仍然是 c=1;d=2;这说明,如果在变量前面加了关键字out,调用函数时的最张结果不会因外部变量的定义而改变,如果要改变函数的

值,只能从函数体内部入手

再看最后 refTest(ref o, ref p);

虽然前面己经赋值

int o = 11, p = 22;

但是他的输出结果还是没有因此因改变

最后,总结一下

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这两个关键字都能够提供相似的功效,其作用也很像C中的指针变量。稍有不同之处是:

  1. 使用Ref型参数时,传入的参数必须先被初始化。而Out则不需要,对Out而言,就必须在方法中对其完成初始化。
  2. 使用Ref和Out时都必须注意,在方法的参数和执行方法时,都要加Ref或Out关键字。以满足匹配。
  3. Out更适合用在需要Return多个返回值的地方,而Ref则用在需要被调用的方法修改调用者的引用的时候。
posted @ 2011-08-24 17:08  一个土豆一棵青菜  阅读(447)  评论(1编辑  收藏  举报