• 你想要返回多个结果吗?来看看吧 :Ref和Out关键字异同


    类型介绍

    在几乎所有的OOP语言中,都存在2种类型的值。

    1. 值类型
    2. 引用类型

    以C#为例:其值类型为sbyte,byte,char,short,ushort,int,uint,long和ulong,float和double,当然还有decimal和bool。而引用类型则是string和object。

    我想说的

    我想说的就是——Ref和Out把我弄糊涂的原因是,当时没有认真的去分析它对不同类型所做出的不同的动作。

    对于值类型。

    使用了Ref和Out的效果就几乎和C中使用了指针变量一样。它能够让你直接对原数进行操作,而不是对那个原数的Copy进行操作。举个小例子:

    using System;namespace ConsoleApplication4
    {
    /// <summary>
    /// Class1 的摘要说明。
    /// </summary>
    class Class1
    {
    /// <summary>
    /// 应用程序的主入口点。
    /// </summary>
    [STAThread]
    static void Main(string[] args)
    {
    int a = 5;
    int b; squareRef(ref a);
    squareOut(
    out b); Console.WriteLine("The a in the Main is: " + a);
    Console.WriteLine(
    "The b in the Main is: " + b);
    }
    static void squareRef(ref int x)
    {
    x
    = x * x;
    Console.WriteLine(
    "The x in the squareRef is: " + x);
    }
    static void squareOut(out int y)
    {
    y
    = 10;
    y
    = y * y;
    Console.WriteLine(
    "The y in the squareOut is: " + y);
    }
    }
    }

      

    显示的结果就是——25 100 25 100。

    这样的话,就达到了和C中的指针变量一样的作用。

    对于引用类型。

    对于引用类型就比较难理解了。

    先要了解到这一层——就是当一个方法接收到一个引用类型的变量的时候,它将获得这个引用(Reference)的一个Copy。由于Ref关键字可以用来向方法传递引用。所以,如果这个功能被误用了——比如:当一个如数组类型的引用对象用关键字Ref传递的时候,被调用的方法实际上已经控制了传递过来的引用本身。这样将使得被调用方法能用不同的对象甚至NULL来代替调用者的原始引用!

    如图。内存地址为2000的变量arrayA中其实存放着数组{1,2,3,4,……}的内存起始地址10000。如果一个方法fun()使用fun( arrayA[] )的话,它将顺利的获得数据10000,但这个10000将放在一个Copy中,不会放到内存的2000位置。而这个时候我们如果使用fun( ref arrayA[] )的话,我们得到的值就是2000啦(也就是说,被调用方法能够修改掉arrayA中的那个引用,使之不再指向10000,甚至可以用NULL来代替10000,这样的话,那个10000地址中的数据可能就要被垃圾回收机制清理了。)

    01 using System;
    02  class TestApp
    03  {
    04  static void outTest(out int x, out int y)
    05  {//离开这个函数前,必须对x和y赋值,否则会报错。
    06  //y = x;
    07  //上面这行会报错,因为使用了out后,x和y都清空了,需要重新赋值,即使调用函数前赋过值也不行
    08  x = 1;
    09  y = 2;
    10  }
    11  static void refTest(ref int x, ref int y)
    12  {
    13  x = 1;
    14  y = x;
    15  }
    16  public static void Main()
    17  {
    18  //out test
    19  int a,b;
    20  //out使用前,变量可以不赋值
    21  outTest(out a, out b);
    22  Console.WriteLine("a={0};b={1}",a,b);
    23  int c=11,d=22;
    24  outTest(out c, out d);
    25  Console.WriteLine("c={0};d={1}",c,d);
    26  //ref test
    27  int m,n;
    28  //refTest(ref m, ref n);
    29  //上面这行会出错,ref使用前,变量必须赋值
    30  int o=11,p=22;
    31  refTest(ref o, ref p);
    32  Console.WriteLine("o={0};p={1}",o,p);
    33  }
    34  }

    运行结果如下:

    下法我解析一下为什么是这样的结果:

    ref是传递参数的地址,out是返回值,两者有一定的相同之处,不过也有不同点。

      - 使用ref前必须对变量赋值,out不用。

      - out的函数会清空变量,即使变量已经赋值也不行,退出函数时所有out引用的变量都要赋值,ref引用的可以修改,也可以不修改。

    在上面的代码中:

    首先看 outTest(out a, out b)的输出结果

    结果是 a=1;b=2;这是在 static void outTest(out int x, out int y)函数定义时的x,y的值

    x=1;y=2;然后就直接传递给a,b了

    接下来再看 outTest(out c, out d);

    这个与 outTest(out a, out b)的不同之处在于

    他前面己给 c,d赋值

    int c = 11, d = 22;

    一般常理来说,这会outTest(out c, out d)的输出值应为 c=11;d=22;

    但是真实结果却仍然是 c=1;d=2;这说明,如果在变量前面加了关键字out,调用函数时的最张结果不会因外部变量的定义而改变,如果要改变函数的

    值,只能从函数体内部入手

    再看最后 refTest(ref o, ref p);

    虽然前面己经赋值

    int o = 11, p = 22;

    但是他的输出结果还是没有因此因改变

    最后,总结一下

    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这两个关键字都能够提供相似的功效,其作用也很像C中的指针变量。稍有不同之处是:

    1. 使用Ref型参数时,传入的参数必须先被初始化。而Out则不需要,对Out而言,就必须在方法中对其完成初始化。
    2. 使用Ref和Out时都必须注意,在方法的参数和执行方法时,都要加Ref或Out关键字。以满足匹配。
    3. Out更适合用在需要Return多个返回值的地方,而Ref则用在需要被调用的方法修改调用者的引用的时候。
  • 相关阅读:
    Packet for query is too large (1986748 > 1048576). You can change this value on the server by 异常
    解决springdatajpa插入大量数据速度慢的问题
    thymeleaf onclick方法向js方法传递参数
    git的使用(扫盲)
    【错误总结】Component template should contain exactly one root element. If you are using v-if on multiple elements, use v-else-if to chain them instead.
    SpringBoot集成Swagger(Swagger的使用),生成接口文档,方便前后端分离开发
    spring中后台接收参数总结
    PTA 03-树3 Tree Traversals Again (25分)
    PTA 03-树2 List Leaves (25分)
    PTA 03-树1 树的同构 (25分)
  • 原文地址:https://www.cnblogs.com/0banana0/p/2152253.html
Copyright © 2020-2023  润新知