在C#中,你在哪里使用参数前面的“ref”?

时间:2009-04-29 20:48:55

标签: c# ref

关于“ref”和“out”参数的定义已经有很多问题,但它们看似糟糕的设计。您是否认为ref是正确的解决方案?

看起来你总能做一些更干净的事情。有人能给我一个例子,说明这将成为问题的“最佳”解决方案吗?

10 个答案:

答案 0 :(得分:11)

在我看来,ref在很大程度上弥补了宣布新的实用工具类型的困难以及对现有信息“处理信息”的困难,这些信息是C#自成立以来为解决问题而迈出的重要一步。 LINQ,泛型和匿名类型。

所以不,我认为不再有很多明确的用例了。我认为这很大程度上是该语言最初设计的遗留物。

我认为在你需要从函数返回某种错误代码以及返回值的情况下仍然有意义(如上所述),但没有别的(所以更大的类型不是非常合理。)如果我在项目中的所有地方都这样做,我可能会为thing-plus-error-code定义一些通用的包装器类型,但在任​​何给定的实例中refout没事。

答案 1 :(得分:9)

嗯,ref通常用于特殊情况,但我不会将其称为冗余或C#的遗留功能。你会看到它(和out)在XNA中使用了很多。在XNA中,Matrix是一个struct,而且是一个相当大的(我相信64个字节),如果你使用ref将其传递给函数以避免复制64个字节,通常是最好的,但只有4或8.专家C#功能?当然。不再使用或表示不好的设计?我不同意。

答案 2 :(得分:7)

一个领域是使用小实用功能,例如:

void Swap<T>(ref T a, ref T b) { T tmp = a; a = b; b = tmp; }  

我在这里看不到任何“更清洁”的选择。当然,这不完全是架构级别。

答案 3 :(得分:4)

P / Invoke是我唯一能想到你必须使用ref或out的地方的地方。在其他情况下,它们可以很方便,但就像你说的那样,通常还有另一种更清洁的方式。

答案 4 :(得分:3)

如果您想要返回多个对象,那么由于某些未知原因,这些对象并不会绑定到一个对象中。

void GetXYZ( ref object x, ref object y, ref object z);

编辑: divo建议使用OUT参数更适合这一点。我不得不承认,他有一点意见。我将把这个答案留在这里作为记录,这是一个不合适的解决方案。在这种情况下,OUT胜过REF。

答案 5 :(得分:1)

我认为最好的用途是那些你经常看到的用途;你需要同时拥有一个值和一个“成功指标”,它不是一个函数的例外。

答案 6 :(得分:1)

的实际用途是创建结构时。 C#中的结构是值类型,因此在按值传递时始终完全复制。如果需要通过引用传递它,例如出于性能原因或者因为函数需要对变量进行更改,则可以使用ref关键字。

我可以看到某人是否有一个包含100个值的结构(显然已经存在问题),您可能希望通过引用传递它以防止100个值复制。那个并返回那个大型结构并写入旧值可能会出现性能问题。

答案 7 :(得分:1)

ref有用的一种设计模式是双向访问者。

假设您有一个Storage类,可用于加载或保存各种基元类型的值。它处于Load模式或Save模式。它有一组称为Transfer的重载方法,这是处理int值的示例。

public void Transfer(ref int value)
{
    if (Loading)
        value = ReadInt();
    else
        WriteInt(value);
}

其他原始类型会有类似的方法 - boolstring等。

然后在需要“转移”的类上,你会写一个这样的方法:

public void TransferViaStorage(Storage s)
{
    s.Transfer(ref _firstName);
    s.Transfer(ref _lastName);
    s.Transfer(ref _salary);
}

同样的单一方法可以加载Storage中的字段,也可以将字段保存到Storage,具体取决于Storage对象所处的模式。

实际上,您只是列出了需要传输的所有字段,因此它非常接近声明性编程而非强制性。这意味着您不需要编写两个函数(一个用于读取,一个用于写入),并且假设我在这里使用的设计是依赖于顺序的,那么确定字段将始终被读取非常方便/以相同的顺序书写。

一般来说,当参数标记为ref时,您不知道该方法是读取它还是写入它,这样您就可以设计一个参数类两个方向,旨在以对称方式调用(即访问方法不需要知道访问者类在哪个方向模式下运行)。

比较:属性+反思

为什么这样做而不是归因于字段并使用反射来自动实现等效的TransferViaStorage?因为有时候反射很慢就会成为一个瓶颈(但总是要确定这一点 - 它几乎不是真的,并且属性更接近于声明性编程的理想)。

答案 8 :(得分:0)

使用“ref”关键字的明显原因是您希望通过引用传递变量。例如,将System.Int32之类的值类型传递给方法并更改其实际值。更具体的用途可能是您想要交换两个变量。

public void Swap(ref int a, ref int b)
{
   ...
}

使用“out”关键字的主要原因是从方法返回多个值。我个人更喜欢将值包装在专门的结构或类中,因为使用out参数会产生相当丑陋的代码。用“out”传递的参数 - 就像“ref” - 通过引用传递。

public void DoMagic(out int a, out int b, out int c, out int d)
{
   ...
}

答案 9 :(得分:0)

当您必须使用'ref'关键字时,有一个明显的例子。如果对象已定义但未在您打算调用的方法范围之外创建,并且您要调用的方法应该'new'来创建它,则必须使用'ref'。例如{object a; Funct(a);} {Funct(object o) {o = new object; o.name = "dummy";}不会对object 'a'做任何事情,也不会在编译或运行时抱怨它。它不会做任何事情。 {object a; Funct(ref a);} {Funct(object ref o) {o = new object(); o.name = "dummy";}会导致'a'成为名为"dummy的新对象。但是如果'new'已经完成,则不需要ref(但如果提供则有效) 。{object a = new object(); Funct(a);} {Funct(object o) {o.name = "dummy";}