C# 中的 ref 关键字

转载 2012年03月23日 15:21:15

对于值类型参数,使用ref关键字,那么传递给方法的将是这个参数的地址,而不是参数的值类型实例的拷贝。
假如一个方法的参数加上ref关键字,那么在调用该方法之前必须首先初始化该参数,被调用方法可以任意选择读取或修改该参数的值。

对于引用类型参数,传递给方法的就是这个参数的地址,所以一般不需要使用ref关键字(除了用new对这个参数重新赋值)。

开始我们的例子之前,需要提前说明几点:

  • C# 中的数据有两种类型:引用类型(reference types)和值类型(value types)。 简单类型(包括int, long, double等)和结构(structs)都是值类型,而其他的类都是引用类型。 简单类型在传值的时候会做复制操作,而引用类型只是传递引用,就像 C++ 中的指针一样。
  • 注意 structs 在 C# 和 C++ 中的区别。在 C++ 中, structs 和类基本相同(except that the default inheritance and default access are public rather than private)。 而在 C# 中,structs 和类有很大的区别。其中最大的区别(我个人觉得,同时也是容易忽略的一个地方)可能就是它是值类型,而不是引用类型。
// cs_ref.cs
using System;
public class MyClass
{
  public static void TestRef(ref char i)
  {
    // The value of i will be changed in the calling method

    i = 'b';
  }

  public static void TestNoRef(char i)
  {
    // The value of i will be unchanged in the calling method
    i = 'c';
  }

  // This method passes a variable as a ref parameter; the value of the
  // variable is changed after control passes back to this method.
  // The same variable is passed as a value parameter; the value of the
  // variable is unchanged after control is passed back to this method.
  public static void Main()
  {
    char i = 'a';    // variable must be initialized

    TestRef(ref i);  // the arg must be passed as ref
    Console.WriteLine(i);
    TestNoRef(i);
    Console.WriteLine(i);
  }
}

下面这段代码是 MSDN 中的例子: 大家很容易看出输出结果是:

 
 
b
b

那么如果把这个例子做一些新的改动,将值类型(这里用的是 char)改成引用类型,程序运行又是什么效果呢?

// ----------------------------------------
// MyClass definition
public class MyClass
{
  public int Value;
}


// ----------------------------------------
// Tester methods

public static void TestRef(ref MyClass m)
{
  m.Value = 10;
}

public static void TestNoRef(MyClass m)
{
  m.Value = 20;
}

public static void TestCreateRef(ref MyClass m)
{
  m = new MyClass();
  m.Value = 100;
}


public static void TestCreateNoRef(MyClass m)
{
  m = new MyClass();
  m.Value = 200;
}

public static void Main()
{
  MyClass m = new MyClass();
  m.Value = 1;

  TestRef(ref m);
  Console.WriteLine(m.Value);


  TestNoRef(m);
  Console.WriteLine(m.Value);

  TestCreateRef(ref m);
  Console.WriteLine(m.Value);

  TestCreateNoRef(m);
  Console.WriteLine(m.Value);
}

 

大家能马上给出正确的答案么?如果能,那看来你对 ref 的用法了解得还是非常不错的。其实如果大家对 C++ 比较熟悉的话,把这段代码换成 C++ 的就好理解的多了。

// ----------------------------------------
// MyClass definition
#pragma once

class MyClass
{
public:
  int Value;
};

typedef MyClass* MyClassPtr;


// ----------------------------------------
// Tester methods
void TestRef(char* i)
{
  *i = 'b';
}

void TestNoRef(char i)
{
  i = 'c';
}


void TestRef(MyClassPtr* m)
{
  (*m)->Value = 10;
}

void TestNoRef(MyClassPtr m)
{
  m->Value = 20;
}

void TestCreateRef(MyClassPtr* m)
{
  delete (*m);
  *m = new MyClass();
  (*m)->Value = 100;
}


void TestCreateNoRef(MyClassPtr m)
{
  m = new MyClass();
  m->Value = 200;
}

int main(int argc, char* argv[])
{
  char c = 'a';


  TestRef(&c);
  printf("%c\n", c);  // output: b
  TestNoRef(c);
  printf("%c\n", c);  // output: b

  MyClassPtr m = new MyClass;
  m->Value = 1;


  TestRef(&m);
  printf("%d\n", m->Value);

  TestNoRef(m);
  printf("%d\n", m->Value);

  TestCreateRef(&m);
  printf("%d\n", m->Value);

  TestCreateNoRef(m);
  printf("%d\n", m->Value);

  delete m;

  return 0;
}

 

这两段分别用 C# 和 C++ 实现的代码的输出结果都是一样的。后面用 MyClass 测试的输出结果是:

 
 
 
 
10
20
100
100

具体的原因相信经过大家的分析应该会很清楚的。另外如果大家有兴趣可以用 structs 再试试,也可以同时对 structs 在 C++ 和 C# 中的区别有进一步的认识。

正确理解C#中的ref关键字

  • 2015年11月09日 21:27
  • 39KB
  • 下载

(C#)方法参数关键字:ref、out、params详解

备注:以下来自MSDN和网络参考,经过整理后的文档 ref(C# 参考) ref 关键字使参数按引用传递。其效果是,当控制权传递回调用方法时,在方法中对参数的任何更改都将反映在该变量中。若要使用 ...

c#中的关键字out是什么意思?ref是什么意思?它们有什么区别?

不知大家有没有遇到过需要一个函数返回多个值的情况。 当写代码要返回多个值的时候,当然可以返回一个数组来实现,但如果遇到需要返回的多个值的类型不同呢?这个时候怎么办? c#中,out关键字和ref关...

C#关键字params, ref, out的用法

第一次使用params这个关键字,发觉这是蛮好用的一种参数形式。 1、params params关键字用在指定参数可变的场合中。在方法声明中加了params关键字后不能再同时定义其他参数,在声明方...

c#中ref和out 关键字

问题:为什么c#中要有ref和out?(而java中没有) 需求假设:现需要通过一个叫Swap的方法交换a,b两个变量的值。交换前a=1,b=2,断言:交换后a=2,b=1。 现编码如下: cl...
  • oemoon
  • oemoon
  • 2012年07月30日 10:32
  • 568

C#参数修饰符(关键字out,ref,params),可选参数

参数修饰符,可选参数的使用

正确理解C#中的ref关键字

http://www.sudu.cn/info/article/articleInfo.php?aId=235727
  • xumuge
  • xumuge
  • 2011年03月09日 09:01
  • 247

C#中的两个关键字ref和out

今天简要的说下C#中的两个关键字,ref和out关键字,其实在之前学习java的时候,还真是没接触过这两个东西,刚开始看到,还觉得,C#怎么就那么多关键字,不过了解后,还是蛮简单的,就是引用的方式来传...

C# 参考之方法参数关键字:params、ref及out

总结: ref和out都对函数参数采用引用传递形式——不管是值类型参数还是引用类型参数,并且定义函数和调用函数时都必须显示生命该参数为ref/out形式。两者都可以使函数传回多个结果。 两者区别:...

C# 参考之方法参数关键字:params、ref及out

如果在为方法声明参数时未使用 ref 或 out,则该参数可以具有关联的值。可以在方法中更改该值,但当控制传递回调用过程时,不会保留更改的值。通过使用方法参数关键字,可以更改这种行为。 params...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:C# 中的 ref 关键字
举报原因:
原因补充:

(最多只允许输入30个字)