您的位置:首页 > 编程语言 > C#

c# 深度解析方法参数的关键字ref

2016-11-08 18:02 288 查看
昨天在垒代码的时候遇到了一个基础没打牢就会暴露的问题。传递给方法的参数为类(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/
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  c# ref