类型介绍
在几乎所有的OOP语言中,都存在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中的指针变量。稍有不同之处是:
- 使用Ref型参数时,传入的参数必须先被初始化。而Out则不需要,对Out而言,就必须在方法中对其完成初始化。
- 使用Ref和Out时都必须注意,在方法的参数和执行方法时,都要加Ref或Out关键字。以满足匹配。
- Out更适合用在需要Return多个返回值的地方,而Ref则用在需要被调用的方法修改调用者的引用的时候。