C#中使用ref和out详解

【摘 要】 传值参数传递的是调用参数的一份拷贝,而传址参数传递的是调用参数的内存地址,该参数在方法内外指向的是同一个存储位置。

  

首先,如果不使用这两个关键字,那是什么样 呢?

看下面的例子:

using System;
  class Test
  {
      static void Swap(ref int x, ref int y)
      {
          int temp = x;
          x = y;
          y = temp;
      }
      static void Swap(int x,int y)
      {
          int temp = x;
          x = y;
          y = temp;
      }
      static void Main()
      {
          int i = 1, j = 2;
          Swap(ref i, ref j);
          Console.WriteLine("i = {0}, j = {1}", i, j);
          Swap(i,j);
          Console.WriteLine("i = {0}, j = {1}", i, j);
      }
  }
  
    程序经编译后执行输出:
  
  i = 2, j = 1
  i = 2, j = 1

这是csdn的一篇帖子上的例子.其实如果放在一起,并不能很容易的看出使用ref和不使用ref的区别.

分开看,就很明显了.

使用ref:

using System;
  class Test
  {
      static void Swap(ref int x, ref int y)
      {
          int temp = x;
          x = y;
          y = temp;
      }

      
      static void Main()
      {
          int i = 1, j = 2;
          Swap(ref i, ref j);
          Console.WriteLine("i = {0}, j = {1}", i, j);           
      }
  }
  
    程序经编译后执行输出:
  
  i = 2, j = 1


不使用:

using System;
  class Test
  {       
      static void Swap(int x,int y)
      {
          int temp = x;
          x = y;
          y = temp;
      }
      static void Main()
      {
          int i = 1, j = 2;
          Swap(i,j);
          Console.WriteLine("i = {0}, j = {1}", i, j);
      }
  }
  
    程序经编译后执行输出:

          i = 1, j = 2;

比较这两个里子很明显,使用了ref关键字的函数调用以后,i和j的值变化了.而没有使用ref的却没有变,为什么呢?

请看下面这段话:

    方法的参数是个值得特别注意的地方。方法的参数传递有四种类型:传值(by value),传址(by reference),输出参数(by output),数组参数(by array)。传值参数无需额外的修饰符,传址参数需要修饰符ref,输出参数需要修饰符out,数组参数需要修饰符params。传值参数在方法调用过程中如果改变了参数的值,那么传入方法的参数在方法调用完成以后并不因此而改变,而是保留原来传入时的值。传址参数恰恰相反,如果方法调用过程改变了参数的值,那么传入方法的参数在调用完成以后也随之改变。实际上从名称上我们可以清楚地看出两者的含义--传值参数传递的是调用参数的一份拷贝,而传址参数传递的是调用参数的内存地址,该参数在方法内外指向的是同一个存储位置。

          所以我们可以看出,平时我们写的函数参数是传值参数(传值参数无需额外的修饰符),不论他是值类型还是引用类型.你可以试这个例子(这个例子是对"不论他是值类型还是引用类型"这句话,):

using System;
  class Test
  {       
      static void Swap(string x,string y)
      {
          string temp = x;
          x = y;
          y = temp;
      }
      static void Main()
      {
          string i = "1", j = "2";
          Swap(i,j);
          Console.WriteLine("i = {0}, j = {1}", i, j);
      }
  }
  
    程序经编译后执行输出:

          i = "1", j = "2";

          注意:string类型是引用类型.

i,j的值并没有改变,说明引用类型作为函数参数时,只要不加ref或者out,那他仍然是传值参数(我一直以为值类型作为函数参数时是传值参数,而引用类型作为函数参数时是传址参数).

.net框架程序设计(修订版)上有这样一句话:ref和out的区别是参数的初始化和参数返回,

可是这里说out是输出参数,那加out是传址参数吗?要不,怎么也会变呢(应该也是吧,可能他只是一个名字,可以理解成输出参数也是传址参数)?

using System;
  class Test
  {
      static void Swap(out int x, out int y)
      {
          int temp = x;
          x = y;
          y = temp;
      }

      
      static void Main()
      {
          int i = 1, j = 2;
          Swap(out i, out j);
          Console.WriteLine("i = {0}, j = {1}", i, j);           
      }
  }

程序经编译后执行输出:
  
  i = 2, j = 1


总结:传值参数传递的是调用参数的一份拷贝,而传址参数传递的是调用参数的内存地址,该参数在方法内外指向的是同一个存储位置.

这句话可以这样说更加明白:传值参数传递的是调用参数的拷贝的地址,该参数在方法内外指向的不是是同一个存储位置,而传址参数传递的是调用参数的内存地址,该参数在方法内外指向的是同一个存储位置,

理解这句话,自然明白为什么会变化或者为什么不变化.

以上是个人的一点理解,肯定有不妥之处,请批评指出.

 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值