C#中ref与out的区别


// 今天遇到一个问题,需要传递一个已经初始化的值,到另外一个函数里递增,然后返回递增后的值。
// 顺便总结一下out与ref的区别
private  void button1_Click( object sender, EventArgs e)
{
     // ref,out都能修改传进来的参数的值。
     int refInt =  100;
    RefValue( ref refInt);
    MessageBox.Show(refInt.ToString());
     //
     int outInt =  100;
    outValue( out outInt,  100);
    MessageBox.Show(outInt.ToString());
}
// ref
public  void RefValue( ref  int refInt)
{
    refInt +=  100; // ref引用不需要初始化
}
// 一个函数可以有多个ref参数
public  void RefValue( ref  int refInt, ref  int rInt)
{
    refInt +=  100; // ref引用不需要初始化
}
// out
public  void outValue( out  int outInt,  int i)
{
    outInt = i +  100; // out引用必须初始化
}
/* 函数不可以有多个out参数
public void outValue(out int outInt, out oInt,int i)
{
    outInt = i + 100;//out引用必须初始化
}
*/
url: http://greatverve.cnblogs.com/archive/2012/02/27/ref-out.html
修正:
/* 函数可以有多个out参数 */
public  void outValue( out  int outInt,  out  int oInt, int i)
{
    outInt = i +  100; // out引用必须初始化
    oInt =  100; // out的值必须在方法里定义。
}
-----------------------------------------------------------------------------

ref和out的区别在C# 中,既可以通过值也可以通过引用传递参数。通过引用传递参数允许函数成员更改参数的值,并保持该更改。若要通过引用传递参数, 可使用ref或out关键字。ref和out这两个关键字都能够提供相似的功效,其作用也很像C中的指针变量。它们的区别是:

1、使用ref型参数时,传入的参数必须先被初始化。对out而言,必须在方法中对其完成初始化。

2、使用ref和out时,在方法的参数和执行方法时,都要加Ref或Out关键字。以满足匹配。

3、out适合用在需要retrun多个返回值的地方,而ref则用在需要被调用的方法修改调用者的引用的时候。

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

方法参数上的 ref 方法参数关键字使方法引用传递到方法的同一个变量。当控制传递回调用方法时,在方法中对参数所做的任何更改都将反映在该变量中。

若要使用 ref 参数,必须将参数作为 ref 参数显式传递到方法。ref 参数的值被传递到 ref 参数。

传递到 ref 参数的参数必须最先初始化。将此方法与 out 参数相比,后者的参数在传递到 out 参数之前不必显式初始化。

属性不是变量,不能作为 ref 参数传递。

如果两种方法的声明仅在它们对 ref 的使用方面不同,则将出现重载。但是,无法定义仅在 ref 和 out 方面不同的重载。

out

方法参数上的 out 方法参数关键字使方法引用传递到方法的同一个变量。当控制传递回调用方法时,在方法中对参数所做的任何更改都将反映在该变量中。

当希望方法返回多个值时,声明 out 方法非常有用。使用 out 参数的方法仍然可以返回一个值。一个方法可以有一个以上的 out 参数。

若要使用 out 参数,必须将参数作为 out 参数显式传递到方法。out 参数的值不会传递到 out 参数。

不必初始化作为 out 参数传递的变量。然而,必须在方法返回之前为 out 参数赋值。

属性不是变量,不能作为 out 参数传递。


网上有很多文章说ref 只传值,out传地址等等这种说法,好像不是非常的准确。以下是我做的实例代码,大家可以去试试:

 public int  RefValue(int i,ref int j)
        {
            int k = j;
            j =222;
            return i+k;
        }

      
        public int OutValue(int i, out int j)
        {
            j = 222;
            return i + j;
        }

        private void cmdRef_Click(object sender, EventArgs e)
        {
            int m = 0;
            MessageBox.Show(RefValue(1, ref m).ToString());
            MessageBox.Show(m.ToString());
        }

        private void cmdOut_Click(object sender, EventArgs e)
        {
            int m;
            MessageBox.Show(OutValue(1, out m).ToString());
            MessageBox.Show(m.ToString());
        }
借网上总结的一句话说,ref是有进有出,而out是只出不进。
--------------------------------------------------------------------

  ref    
    
  通常我们向方法中传递的是值.方法获得的是这些值的一个拷贝,然后使用这些拷贝,当方法运行完毕后,这些拷贝将被丢弃,而原来的值不将受到影响.此外我们还有其他向方法传递参数的形式,引用(ref)和输出(out).
    
  有时,我们需要改变原来变量中的值,这时,我们可以向方法传递变量的引用,而不是变量的值.引用是一个变量,他可以访问原来变量的值,修改引用将修改原来变量的值.变量的值存储内存中,可以创建一个引用,他指向变量在内存中的位置.当引用被修改时,修改的是内存中的值,因此变量的值可以将被修改.当我们调用一个含有引用参数的方法时,方法中的参数将指向被传递给方法的相应变量,因此,我们会明白,为什么当修改参数变量的修改也将导致原来变量的值.
    
  创建参数按引用传递的方法,需使用关键字ref.例;

using  System;
    
class  gump
    {
public  double  square( ref  double  x)
{
  x
= x * x;
  
return  x;
}
    }

    
class  TestApp
    {
public  static  void  Main()
{
  gump doit
= new  gump();

  
double  a = 3 ;
  
double  b = 0 ;

  Console.WriteLine(\
" Before square->a={0},b={1}\",a,b);

  b
= doit.square( ref  a);
  Console.WriteLine(\
" After square->a={0},b={1}\",a,b);
}
    }

  通过测试,我们发现,a的值已经被修改为9了.

  out
    
  通过指定返回类型,可以从方法返回一个值,有时候(也许还没遇到,但是我们应该有这么个方法),需要返回多个值,虽然我们可以使用ref来完成,但是C#专门提供了一个属性类型,关键字为out.介绍完后,我们将说明ref和out的区别.
  
通过使用out关键字,我们改变了三个变量的值,也就是说out是从方法中传出值.

using  System;
    
class  gump
    {
public  void  math_routines( double  x, out  double  half, out  double  squared, out  double  cubed)
// 可以是:public void math_routines( // ref double x,out double half,out double squared,out double cubed)
// 但是,不可以这样:public void math_routines(out double x,out double half,out double squared,out double cubed),对本例来说,因为输出的值要靠x赋值,所以x不能再为输出值
{
  half
= x / 2 ;
  squared
= x * x;
  cubed
= x * x * x;
}
    }

    
class  TestApp
    {
public  static  void  Main()
{
    gump doit
= new  gump();

    
double  x1 = 600 ;
    
double  half1 = 0 ;
    
double  squared1 = 0 ;
    
double  cubed1 = 0 ;
    [Page]
    
/*
     double x1=600;
     double half1;
     double squared1;
     double cubed1;
    
*/

    Console.WriteLine(\
" Before method->x1={0}\",x1);
    Console.WriteLine(\ " half1={0}\",half1);    Console.WriteLine(\"squared1={0}\",squared1);
    Console.WriteLine(\ " cubed1={0}\",cubed1);



    doit.math_rountines(x1,
out  half1, out  squared1, out  cubed1);
    
    Console.WriteLine(\
" After method->x1={0}\",x1);
    Console.WriteLine(\ " half1={0}\",half1);
    Console.WriteLine(\ " squared1={0}\",squared1);
    Console.WriteLine(\ " cubed1={0}\",cubed1);
}
    }

  我们发现,ref和out似乎可以实现相同的功能.因为都可以改变传递到方法中的变量的值.但是,二者本质本质的区别就是,ref是传入值,out是传出值.在含有out关键字的方法中,变量必须由方法参数中不含out(可以是ref)的变量赋值或者由全局(即方法可以使用的该方法外部变量)变量赋值,out的宗旨是保证每一个传出变量都必须被赋值.
    

  上面代码中被/**/注释掉的部分,可以直接使用.也就是说,在调用方法前可以不初始化变量.但是\"x1\"是要赋值的,否则要报错.而ref参数,在传递给方法时,就已经是还有值的了,所以ref侧重修改.out侧重输出. 
-----------------------------------------------------------------------------

关于c#方法返回多个值的几个方法

在实际编程中,我们会经常遇到在同一个c#方法中返回多个值的情况,根据本人和经验,现总结了如下两种方法:

1.用out 声明返回值

例public void currentMonthCount(out int monthCome ,out DataTable dtTable)

其中monthCome   dtTable是两个不同类型的返回值.

此外,还可以既有多个返回值,还可以同时加入传入参数.

例:public void query(out int come,out   DataTable dtTable, DateTime startDate, DateTime endDate)

其中come    dtTable为返回值,startDate,endDate为传入参数.

同时注意,前面必须用void修饰,否则会出错.

调用时:须首先定义返回值的类型,如

调用public void currentMonthCount(out int monthCome ,out DataTable dtTable)这个方法

首先定义:int monthCome;

DataTable dtTable;

cuurentMonthCount(out monthCome,out dtTable);

this.lable1.text=monthCome.toString();

就能得到返回值了.

2,定义object 型函数

例:用一个参数作为条件返回多个不同的值,

public object MakeConnectionMethod(string sql,   executeMethod execMethod)

{

//返回dataSet
             if (strConnection != "" && execMethod == executeMethod.execute_DataSet)
             {

                ..........

                 return mydataset1;
             }


             //添加、删除、修改
             if (strConnection != "" && execMethod == executeMethod.execute_NoneQuery)
             {
                 ........

                 }
                 return isSucceed;
             }

             //返回关键字第一行记录的arraylist。
             if (strConnection != "" && execMethod == executeMethod.execute_Linekey)
             {
                 .....

                 }
                 return retString;
             }
             return null;
         }

调用时根据不同的参数,可以返回不同的值

 

-------------------------------------------------------------------------------------
一个函数返回多个值 C#
Public  void GetValues( string UserID, string PassWord, out  string  string UserName, out  bool flag)
{
       // 在使用一些判断为返回值赋值时,最好先在判断外赋上初始值,否则易报“离开当前方法之前必须对out 参数赋值”错误
 
      UserName =  "";
      flag =  true;
 
       if(UserID ==  1 && Password ==  " 123 ")
      {
          UserName =  " 小白 ";
      }
       else
      {
          flag =  false;
      }
}

private  void btnLogin_Click( object sender, System.Web.UI.ImageClickEventArgs e)
{
       // 先对返回的参数进行声明
       string Name;
       string UserName;
       bool Flag;

      GetValues( " 1 "" 123 ", out Name,  out Flag);
         
       // 此时Name和Flag的值已返回
      
// Name = "小白"
      
// Flag = true

       if(Flag ==  true)
      {
            UserName = Name;
             // UserName被赋值为小白
      }
}

out 关键字会导致参数通过引用来传递。这与 ref 关键字类似,不同之处在于 ref 要求变量必须在传递之前进行初始化。若要使用 out 参数,方法定义和调用方法都必须显式使用 out 关键字。尽管作为 out 参数传递的变量不需要在传递之前进行初始化,但需要调用方法以便在方法返回之前赋值。ref 和 out 关键字在运行时的处理方式不同,但在编译时的处理方式相同。因此,如果一个方法采用 ref 参数,而另一个方法采用 out 参数,则无法重载这两个方法。 

参考MSDN


免责声明!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系本站邮箱yoyou2525@163.com删除。



 
粤ICP备18138465号  © 2018-2025 CODEPRJ.COM