ref和out有什么区别?

解决方案 »

  1.   

    ref 是按地址传值
    out ,,
      

  2.   

    在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 参数传递。 
      

  3.   

    ref ,out 其实都可以说是引用传递,区别于值传递。
    引用传递 的时候 调用函数对参数的修改会反映到原来的传入的参数上,
    但值传递则不会改变。传递的只是原来变量的 副本。ref : 传递前必须初始化,
    out:不必初始化,就算初始化,也会被无视,
    out 作为输出参数,可以解决函数只有一个返回值的问题,
      

  4.   

    也就是说ref类似于c语言中的指针。 using System;class Account
    {
     private int balance = 0;  //字段
     public int Balance   //属性
     {
      get { return balance; }
      set { balance = value;}
     }
     public void Deposit(int n)
     { this.balance += n; } public void WithDraw(int n)
     { this.balance -= n; }
    }class Client
    {
     public static void Main()
     {
      Account a = new Account();
      a.Balance = 1000; // 可以读写属性,因为属性Balance是public型的
      //a.balance = 1000; //不可以读写字段,因为字段balance是private型的  a.WithDraw(500);
      a.Deposit(2000);
      Console.WriteLine("before Method call: a.balance = {0}", a.Balance);
      
      int x = 1;
      int y = 100;
      int z;
      AMethod(x, ref a.balance ,out z);  //ref a.balance 正确,a.balance是字段变量
      //AMethod(x, ref a.Balance ,out z);  //ref a.Balance 错误,a.Balance是属性
      y = a.balance;
      Console.WriteLine("After  Method Call : x = {0}, y = {1}, z = {2}", x, y, a.balance);
      AMethod(x, ref y ,out a.balance);  //out a.balance 正确,a.balance是字段变量
      //AMethod(x, ref y ,out a.Balance);  //ref a.Balance 错误,a.Balance是字段变量
      z = a.balance;
      Console.WriteLine("After  Method Call : x = {0}, y = {1}, z = {2}", x, y, a.balance);
      AMethod(x, ref y ,out z);
      Console.WriteLine("After  Method Call : x = {0}, y = {1}, z = {2}", x, y, z);
     } public static void AMethod(int x, ref int y, out int z)
     {
      x = 7;
      y = 8;
      z = 9;
     }
    }输出结果:---------- C# Run ----------
    before Method call: a.balance = 2500
    After  Method Call : x = 1, y = 8, z = 8
    After  Method Call : x = 1, y = 8, z = 9
    After  Method Call : x = 1, y = 8, z = 9输出完成 (耗时: 0 秒) - 正常终止
      

  5.   

    ref和out都是按引用传递,不同的是ref在传递前需要显式初始化
      

  6.   

    简而言之:
    ref : 传递前必须初始化, 
    out:不必初始化
      

  7.   

    这种问题,一般google或baidu都能找到~
      

  8.   

    ref 有初值可理解为双向,
    out 可不初始化。
    我感觉这两个东西完全可以不用
      

  9.   

    一个是引用参数,一个是输入参数。
    使用ref型参数时,传入的参数必须先被初始化。对out而言,必须在方法中对其完成初始化。
    完成初始化的地方有不同。C#中对他们两个赋予的功能是不一样的:
    out适合用在需要多个输出结果或者值的地方,而ref则用在需要被调用的方法修改调用者的引用的时候。
    还有就是属性不能作为out传入。
      

  10.   

    ref 是按地址传值 
    out 也是按地址传值,区别在于ref的变量要赋初值,而out变量则可以先不赋初值
      

  11.   

    out = ref + 方法体内所有路径必须初始化 + 调用者不需要初始化
    out和ref的区别仅仅在某些编译器(例如c#)会区别对待,但是某些编译器(例如vb.net)中两者没有区别
    并且两者在函数签名方面也没有区别,例如下面的重载无法通过编译:
    void F(ref int i)
    {
    //...
    }
    void F(out int i)
    {
    //...
    }
      

  12.   

    使用ref型参数时,传入的参数必须先被初始化。对out而言,必须在方法中对其完成初始化。
    使用ref和out时,在方法的参数和执行方法时,都要加Ref或Out关键字。以满足匹配。
    out适合用在需要retrun多个返回值的地方,而ref则用在需要被调用的方法修改调用者的引用的时候。
        class TestApp
        {
            static void outTest(out int x, out int y)
            {
                x = 1;
                y = 2;
            }
            static void refTest(ref int x, ref int y)
            {
                x = 1;
                y = x;
            }
            public static void Main()
            {
                int a, b;
                outTest(out a, out b);
                Console.WriteLine("a={0};b={1}", a, b);
                int c = 11, d = 22;
                outTest(out c, out d);
                Console.WriteLine("c={0};d={1}", c, d)            int o = 11, p = 22;
                refTest(ref o, ref p);
                Console.WriteLine("o={0};p={1}", o, p);
            }
        }
      

  13.   

    balance 字段为private,在Client类的AMethod方法中无法调用吧,,,
      

  14.   


    赞成. 最开始楼主自己也说了 a.balance 是 private的 , 后来的代码实在是自相矛盾啊...
    楼主貌似有点糊涂...
      

  15.   


    赞成. 最开始楼主自己也说了 a.balance 是 private的 , 后来的代码实在是自相矛盾啊...
    楼主貌似有点糊涂...
    (之前本来想引用18楼的, 结果引用成了17楼了,故在此更正一下。)
      

  16.   

    想像成你去饭店买饭,厨师要给你做饭,就得有原材料,材料就得由饭店外获得,要从外面买进来,这个时候就是用到ref的时候,买进来的原材料已经给初始化了,是现成的了。运用到炒饭这个方法中可以直接使用。
    饭炒好后,你将饭带出饭店,这个时候就如同用到out,你带出去的饭,需要在炒饭的这个方法体中进行初始化,你带出去的饭即是返回的结果。