zoukankan      html  css  js  c++  java
  • c# 深度解析方法参数的关键字ref(原创)

    昨天在垒代码的时候遇到了一个基础没打牢就会暴露的问题。传递给方法的参数为类(class)时,在方法中所做的修改赋值不一定会最终改变到原始的变量上。

    举一个例子,如果一个方法Action(List<int> lst),在方法里面对lst做了很多操作,包括add,remove,new,add等等。传入变量List<int> input,方法执行完之后,input可能被执行了add,remove,但是new以后的任何操作都没有保留。这是为什么呢?最开始学习.net基础的时候就知道,引用类型,传递给方法的是引用的地址,而不是实际数值。那为什么会部分的操作被保留了出来,而部分又没有执行呢?

    用代码来分析此案例。

    static void Main(string[] args)
    {    
        int i = 0;
        int refI = 0;
        List<int> list = new List<int>() { 0, 1, 2 };
        List<int> refList = new List<int>() { 0, 1, 2 };
        TestStruct(i);
        TestRefStruct(ref refI);
        TestClass(list);
        TestRefClass(ref refList);
        Console.WriteLine("i: {0}\r\nrefI: {1}\r\nlist: {2}\r\nrefList: {3}", i, refI,
          list.Select(x => x.ToString()).Aggregate((x, y) => x + "," + y),
          refList.Select(x => x.ToString()).Aggregate((x, y) => x + "," + y));
        Console.ReadKey();
    }
    
    static void TestStruct(int input)
    {
        input = 10;
    }
    
    static void TestRefStruct(ref int input)
    {
        input = 10;
    }
    
    static void TestClass(List<int> input)
    {
        input.Add(5);
        input = new List<int>();
        input.Add(10);
    }
    
    static void TestRefClass(ref List<int> input)
    {
        input.Add(5);
        input = new List<int>();
        input.Add(10);
    }

    调试程序,最后输出

    i: 0
    refI: 10
    list: 0,1,2,5
    refList: 10

    在函数TestStruct中,传入一个值类型的参数,没有对传递进去的参数i做任何的修改。

    在函数TestRefStruct中,传入值类型的参数,通过引用传递参数ref,函数中对input进行的任何改变都影响到了refI上,所做的编辑修改全部保留过来。最终refI的值为10。

    在函数TestClass中,传入一个引用类型的参数,在函数中,对input重新赋值之前所做的修改都保留了下来,影响了list的值。而在对input重新赋值之后的所有修改编辑,都和list没有任何关联了。

    在函数TestRefClass中,传入一个引用类型的参数,同时,参数前面加上ref的约束,函数中,对input进行的任何编辑都影响了refList。最终refList的值为new List<int>{ 10 }。

    有一定.net基础的人都可以很清晰的理解第一、二和第四种情况。但是第三种情况常常会给我们留下陷阱。

    如何理解和正确的对待函数传递的参数为引用类型的情况?我的理解是:第三种情况下,传递给函数的变量A提供的是一个引用地址,函数会自动生成一个变量B,同时用传递进来的引用地址对这个变量B赋值,这时,传递进来的变量A和函数内调用的变量B共一个引用地址,所做的修改会同步的影响另一个参数。如果在函数内部,出现了一个input = new List<int>();的语句。这时,变量B会重新赋值到另一个引用地址。那么,从此之后,变量B与变量A再没有关联,对变量B所做的任何修改将不影响变量A。

    下面模拟代码呈现。

    List<int> A = new List<int>() { 0, 1, 2 };//传递给函数的变量。
    {//进入函数
        List<int> B = A;//函数执行后,自动生成B,同时用A对B赋值。
        B.Add(5);//由于他们是引用类型,共一个引用地址,所做修改相互影响。此时A的值也一起改变。
        B = new List<int>();//对B重新赋值,指向另一个引用地址。与A无关。
        B.Add(10);//A不变。
    }//出函数,B释放,A继续存在。

    个人理解。如果不足请补充。

    笔者原创。转载请注明出处:http://www.cnblogs.com/icyJ/

  • 相关阅读:
    第十六章 课程总复习
    第四章 数据类型及数据库基本操作
    第二章.图形化管理工具
    第十三章 指导学习:人机猜拳
    洛谷 P4396 (离散化+莫队+树状数组)
    洛谷 P1351 (枚举)
    洛谷P5322 (BJOI 2019) DP
    P3376 网络最大流模板(Dinic + dfs多路增广优化 + 炸点优化 + 当前弧优化)
    洛谷 P2176(最短路)
    HDU 6556 (2018CCPC吉林 B题)
  • 原文地址:https://www.cnblogs.com/icyJ/p/ref.html
Copyright © 2011-2022 走看看