简介
在 C#
中,in、ref
和 out
是用于修改方法参数传递方式的关键字,它们决定了参数是按值传递还是按引用传递,以及参数是否必须在传递前初始化。
基本语义对比
修饰符 | 传递方式 | 可读写性 | 必须初始化 | 调用前必须赋值 | 典型场景 |
---|---|---|---|---|---|
ref |
引用传递 | 可读可写 | 需先在调用前初始化 | 是 | 修改调用者变量;传大对象避免拷贝 |
in |
只读引用传递 | 只读(不能赋值) | 需先在调用前初始化 | 是 | 传递大值类型以避免拷贝 |
out |
引用传递 | 必须在方法体内赋值 | 调用前可未初始化 | 否 | 返回多个值 |
示例用法
ref:引用传递,可读可写
csharp
void Increment(ref int x)
{
x++; // 修改调用者的值
}
int a = 5;
Increment(ref a);
Console.WriteLine(a); // 输出 6
-
调用前:
a
必须已被赋值; -
语义:方法内对参数的任何写操作会直接反映到调用者;
-
适用:需要在方法中"回写"数据,或对大结构体避免复制(如
ref struct
)。
out:输出参数,必须在方法内赋值
csharp
bool TryParse(string s, out int result)
{
if (int.TryParse(s, out var tmp))
{
result = tmp;
return true;
}
result = 0;
return false;
}
int value;
if (TryParse("123", out value))
Console.WriteLine(value); // 输出 123
-
调用前:
value
可未初始化; -
方法体内:必须为
out
参数赋值(否则编译不通过); -
语义:专门用于"输出"数据或多返回值场景;
-
C# 7+
支持声明式out
变量:if (int.TryParse(s, out var result)) ...。
in:只读引用传递
csharp
public struct BigStruct
{
public long A, B, C, D;
public long Sum() => A + B + C + D;
}
void PrintSum(in BigStruct bs)
{
// bs = new BigStruct(); // 编译错误:不能修改 in 参数
Console.WriteLine(bs.Sum());
}
var big = new BigStruct { A=1, B=2, C=3, D=4 };
PrintSum(in big);
-
调用前:参数
big
必须初始化; -
方法内:只能读取(编译器禁止赋值或调用会改变其字段的成员);
-
语义:避免对大值类型做整拷贝,同时保证安全只读;
-
性能:适用于 16 字节以上的值类型以减少栈/寄存器拷贝开销;
-
限制:不能与可变成员、属性
setter、ref
局部变量一起使用。
最佳实践
优先使用返回值
csharp
// 避免使用 out
(bool success, int result) = TryParseBetter("123");
大型结构体使用 in
csharp
void ProcessLargeData(in BigStruct data) { ... }
ref 用于需要修改原始值的情况
csharp
void UpdatePosition(ref Vector3 position) { ... }
out 用于需要返回多个值的场景
csharp
bool TryGetValue(string key, out object value)
避免引用类型使用 in/ref
csharp
// 通常不需要 - 引用类型已经通过引用传递
void ProcessList(ref List<int> items) { ... }
差异对比
内存与性能
-
传值(无修饰符)会复制值类型(大对象拷贝成本高);
-
ref/out/in
都传地址,避免复制; -
in
保证只读,编译器可做额外优化。
重载差异
可以同时定义带不同修饰符的方法:
csharp
void Foo(int x) { }
void Foo(ref int x) { }
void Foo(in int x) { }
void Foo(out int x) { x = 0; }
调用时必须明确修饰符:Foo(a), Foo(ref a), Foo(in a), Foo(out a)
泛型约束
C# 7.3+
支持泛型中使用 in/ref
修饰符:
csharp
void Process<T>(in T item) where T : unmanaged { ... }
总结
-
ref
:双向修改,需初始化,适合状态更新。 -
out
:输出参数,无需初始化,方法必须赋值,适合返回多个值。 -
in
:只读引用,需初始化,适合大结构体性能优化。