在 C# 中既可以通过值也可以通过引用传递参数通过引用传递参数允许函数成员(方法属性索引器运算符和构造函数)更改参数的值并保持该更改若要通过引用传递参数请使用 ref 或 out 关键字为简单起见本主题的示例中只使用了 ref 关键字有关 ref 和 out 之间的差异的信息请参见使用 ref 和 out 传递数组
本主题包括下列章节
传递值类型参数
传递引用类型参数
它还包括以下示例
示例 | 演示 | 是否使用 ref 或 out | 通过值传递值类型否
通过引用传递值类型是
交换值类型(两个整数)是
通过值传递引用类型否
通过引用传递引用类型是
交换引用类型(两个字符串)是
传递值类型参数
值类型变量直接包含其数据这与引用类型变量不同后者包含对其数据的引用因此向方法传递值类型变量意味着向方法传递变量的一个副本方法内发生的对参数的更改对该变量中存储的原始数据无任何影响如果希望所调用的方法更改参数值必须使用 ref 或 out 关键字通过引用传递该参数为了简单起见以下示例使用 ref
示例 通过值传递值类型
下面的示例演示通过值传递值类型参数通过值将变量 myInt 传递给方法 SquareIt方法内发生的任何更改对变量的原始值无任何影响
//PassingParamscs
usingSystem;
classPassingValByVal
{
staticvoidSquareIt(intx)
//Theparameterxispassedbyvalue
//ChangestoxwillnotaffecttheoriginalvalueofmyInt
{
x*=x;
ConsoleWriteLine(Thevalueinsidethemethod:{}x);
}
publicstaticvoidMain()
{
intmyInt=;
ConsoleWriteLine(Thevaluebeforecallingthemethod:{}
myInt);
SquareIt(myInt);//PassingmyIntbyvalue
ConsoleWriteLine(Thevalueaftercallingthemethod:{}
myInt);
}
}
输出
The value before calling the method:
The value inside the method:
The value after calling the method:
代码讨论
变量 myInt 为值类型包含其数据(值 )当调用 SquareIt 时myInt 的内容被复制到参数 x 中在方法内将该参数求平方但在 Main 中myInt 的值在调用 SquareIt 方法之前和之后是相同的实际上方法内发生的更改只影响局部变量 x
示例 通过引用传递值类型
下面的示例除使用 ref 关键字传递参数以外其余与示例 相同参数的值在调用方法后发生更改
//PassingParamscs
usingSystem;
classPassingValByRef
{
staticvoidSquareIt(refintx)
//Theparameterxispassedbyreference
//ChangestoxwillaffecttheoriginalvalueofmyInt
{
x*=x;
ConsoleWriteLine(Thevalueinsidethemethod:{}x);
}
publicstaticvoidMain()
{
intmyInt=;
ConsoleWriteLine(Thevaluebeforecallingthemethod:{}
myInt);
SquareIt(refmyInt);//PassingmyIntbyreference
ConsoleWriteLine(Thevalueaftercallingthemethod:{}
myInt);
}
}
输出
The value before calling the method:
The value inside the method:
The value after calling the method:
代码讨论
本示例中传递的不是 myInt 的值而是对 myInt 的引用参数 x 不是 int 类型它是对 int 的引用(本例中为对 myInt 的引用)因此当在方法内对 x 求平方时实际被求平方的是 x 所引用的项myInt
示例 交换值类型
更改所传递参数的值的常见示例是 Swap 方法在该方法中传递 x 和 y 两个变量然后使方法交换它们的内容必须通过引用向 Swap 方法传递参数否则方法内所处理的将是参数的本地副本以下是使用引用参数的 Swap 方法的示例
staticvoidSwapByRef(refintxrefinty)
{
inttemp=x;
x=y;
y=temp;
}
调用该方法时请在调用中使用 ref 关键字如下所示
SwapByRef (ref i ref j);
传递引用类型参数
引用类型的变量不直接包含其数据它包含的是对其数据的引用当通过值传递引用类型的参数时有可能更改引用所指向的数据如某类成员的值但是无法更改引用本身的值也就是说不能使用相同的引用为新类分配内存并使之在块外保持若要这样做请使用 ref(或 out)关键字传递参数为了简单起见以下示例使用 ref
示例 通过值传递引用类型
下面的示例演示通过值向 Change 方法传递引用类型的参数 myArray由于该参数是对 myArray 的引用所以有可能更改数组元素的值但是试图将参数重新分配到不同的内存位置时该操作仅在方法内有效并不影响原始变量 myArray
//PassingParamscs
//Passinganarraytoamethodwithouttherefkeyword
//ComparetheresultstothoseofExample
usingSystem;
classPassingRefByVal
{
staticvoidChange(int[]arr)
{
arr[]=;//Thischangeaffectstheoriginalelement
arr=newint[]{};//Thischangeislocal
ConsoleWriteLine(Insidethemethodthefirstelementis:{}arr[]);
}
publicstaticvoidMain()
{
int[]myArray={};
ConsoleWriteLine(InsideMainbeforecallingthemethodthefirstelementis:{}myArray[]);
Change(myArray);
ConsoleWriteLine(InsideMainaftercallingthemethodthefirstelementis:{}myArray[]);
}
}
输出
Inside Main before calling the method the first element is:
Inside the method the first element is:
Inside Main after calling the method the first element is:
代码讨论
在上个示例中数组 myArray 为引用类型在未使用 ref 参数的情况下传递给方法在此情况下将向方法传递指向 myArray 的引用的一个副本输出显示方法有可能更改数组元素的内容(从 改为 )但是在 Change 方法内使用 new 运算符分配新的内存部分将使变量 arr 引用新的数组因此这之后的任何更改都不会影响原始数组 myArray(它是在 Main 内创建的)实际上本示例中创建了两个数组一个在 Main 内一个在 Change 方法内
示例 通过引用传递引用类型
本示例除在方法头和调用中使用 ref 关键字以外其余与示例 相同方法内发生的任何更改都会影响调用程序中的原始变量
//PassingParamscs
//Passinganarraytoamethodwiththerefkeyword
//ComparetheresultstothoseofExample
usingSystem;
classPassingRefByRef
{
staticvoidChange(refint[]arr)
{
//Bothofthefollowingchangeswillaffecttheoriginalvariables:
arr[]=;
arr=newint[]{};
ConsoleWriteLine(Insidethemethodthefirstelementis:{}arr[]);
}
publicstaticvoidMain()
{
int[]myArray={};
ConsoleWriteLine(InsideMainbeforecallingthemethodthefirstelementis:{}myArray[]);
Change(refmyArray);
ConsoleWriteLine(InsideMainaftercallingthemethodthefirstelementis:{}myArray[]);
}
}
输出
Inside Main before calling the method the first element is:
Inside the method the first element is:
Inside Main after calling the method the first element is:
代码讨论
方法内发生的所有更改都影响 Main 中的原始数组实际上使用 new 运算符对原始数组进行了重新分配因此调用 Change 方法后对 myArray 的任何引用都将指向 Change 方法中创建的五个元素的数组
示例 交换两个字符串
交换字符串是通过引用传递引用类型参数的很好的示例本示例中str 和 str 两个字符串在 Main 中初始化并作为由 ref 关键字修饰的参数传递给 SwapStrings 方法这两个字符串在该方法内以及 Main 内均进行交换
//PassingParamscs
usingSystem;
classSwappinStrings
{
staticvoidSwapStrings(refstringsrefstrings)
//Thestringparameterxispassedbyreference
//Anychangesonparameterswillaffecttheoriginalvariables
{
stringtemp=s;
s=s;
s=temp;
ConsoleWriteLine(Insidethemethod:{}{}ss);
}
publicstaticvoidMain()
{
stringstr=John;
stringstr=Smith;
ConsoleWriteLine(InsideMainbeforeswapping:{}{}
strstr);
SwapStrings(refstrrefstr);//Passingstringsbyreference
ConsoleWriteLine(InsideMainafterswapping:{}{}
strstr);
}
}
输出
Inside Main before swapping: John Smith
Inside the method: Smith John
Inside Main after swapping: Smith John
代码讨论
本示例中需要通过引用传递参数以影响调用程序中的变量如果同时从方法头和方法调用中移除 ref 关键字则调用程序中不会发生任何更改