C#.NET in、out、ref详解

简介

C# 中,in、refout 是用于修改方法参数传递方式的关键字,它们决定了参数是按值传递还是按引用传递,以及参数是否必须在传递前初始化。

基本语义对比

修饰符 传递方式 可读写性 必须初始化 调用前必须赋值 典型场景
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:只读引用,需初始化,适合大结构体性能优化。

相关推荐
玩泥巴的6 小时前
.NET驾驭Word之力:基于规则自动生成及排版Word文档
c#·word·.net·com互操作
专注VB编程开发20年6 小时前
VB.NET多线程排序算法实现:LINQ与正则表达式方法
排序算法·.net·linq
SunnyDays10117 小时前
C# 实现高保真 Excel 转 PDF(无需 Office 环境)
经验分享·c#·excel转pdf
攻城狮CSU7 小时前
C# 数据加载专题 之泛型序列化
java·servlet·c#
爱编程的鱼7 小时前
C# 参数详解:从基础传参到高级应用
开发语言·microsoft·c#
流水线上的指令侠9 小时前
使用C#写微信小程序后端——电商微信小程序
微信小程序·小程序·c#·visual studio
gc_229910 小时前
C#编写的WebApi接口直接返回byte数组引发的问题
c#·byte数组
EQ-雪梨蛋花汤11 小时前
【踩坑记录】从“正在还原所需的工具包”说起:一次 .NET 工程包还原失败的完整排查实录
.net
喵叔哟13 小时前
63.【.NET8 实战--孢子记账--从单体到微服务--转向微服务】--新增功能--预算告警
运维·微服务·.net
专注VB编程开发20年13 小时前
VB.NET2003和VB2008可以导入VB6项目
.net·vb.net·vb6·vb2008