在C语言中,对值类型的变量进行交换工作的时候,如果没有传递地址的话,那么函数结束以后,还是原来的值。这个时候,就需要在函数的参数列表中声明指针,而传递指针,则需要调用函数的时候,&去地址。虽然不知道C#的ref具体内部工作机制,但是,我觉得就好比是C语言的&去地址操作,传递的是一个引用变量,而不是一个值类型变量的副本。不同的是,C#中方法的参数列表中,还是使用ref来修饰变量。
static void Swap(ref int num1, ref int num2)
{
int numTemp = num1;
num1 = num2;
num2 = numTemp;
}
static void Main()
{
int num1 = 4;
int num2 = 5;
// 输出交换之前的数值
Console.WriteLine($"{num1}, {num2}");
// 交换
Swap(ref num1, ref num2);
// 输出交换之后的数值
Console.WriteLine($"{num1}, {num2}");
在Main方法之中调用方法之前,需要初始化变量,才能使用ref传递变量给方法。不同于out,ref修饰的变量是由外部传递到方法内部的。而out变量则是在方法内部赋值,然后才换到方法外部的。
对于out与ref,以及后面的in修饰符,类似史莱姆中的狼牙群族,一人命名,全体升级。额,意思就是,这几个修饰符不能当作方法重载的判断依据。但是,有没有修饰符却可以成为方法重载的依据。
// https://docs.microsoft.com/en-us/dotnet/csharp/language-reference/keywords/ref
class CS0663_Example
{
// Compiler error CS0663: "Cannot define overloaded
// methods that differ only on ref and out".
public void SampleMethod(out int i) { }
public void SampleMethod(ref int i) { }
}
class RefOverloadExample
{
public void SampleMethod(int i) { }
public void SampleMethod(ref int i) { }
}
C#7中,可以返回ref修饰的变量。使用一个用ref修饰的方法,返回一个ref修饰的变量,把ref的操作流程反过来了,由方法内部返回一个引用。当然,对于out来说有一个下划线可以忽略掉out。那么,对于ref来说,也可以用'_'来忽略掉一些引用变量吗?应该是不行的,因为对ref来说,只有传递过去了,才有用处的。