在讲C#参数传递之前,我们先简单讨论下 c#中值类型和引用类型的定义以及区别,有助于我们更好的理解参数传递。

我们从内存的角度来简单讨论下值类型和引用类型的区别。我们都知道值类型存储在栈上,引用类型分别在栈和托管堆上。如下图:

我们通过例子来看下 值类型和引用类型存储结构不同有哪些区别:

定义一个类 (引用类型)

View Code
1 public class Student2     {3         public int Age { get; set; }4 5         public void Say()6         {7             Console.WriteLine("我的年龄是:"+Age);8         }9     }

 

定义一个结构(值类型)

View Code
1 public struct StructStudent2     {3         public int Age { get; set; }4 5         public void Say()6         {7             Console.WriteLine("我的年龄是:" + Age);8         }9     }

 

在控制台输出定义如下代码:

View Code
1  static void Main(string[] args) 2         { 3             Student stu1 = new Student { Age=20}; 4             Student stu2 = new Student(); 5             stu2 = stu1; 6             stu2.Age = 30; 7             stu1.Say(); 8  9             stu2.Say();10         }

输出结果:

 

我们将类换成结构,在看看结果如何

定义代码:

View Code
1 static void Main(string[] args)2         {3             StructStudent stu1 = new StructStudent { Age = 20 };4             StructStudent stu2 = new StructStudent();5             stu2 = stu1;6             stu2.Age = 30;7             stu1.Say();8             stu2.Say();9         }

输出结果:

 

 

我们会发现同样的代码 但是结果不一样,其实 每个变量 都有其堆栈,不同的变量不能共用一个堆栈地址。直接上图

 

由图 我们可以看出 对于值类型 stu2的age属性更改不会影响 stu1 对于引用类型 stu1 和stu2 在栈上的地址是不同的。stu1=stu2 。这是2个相等的变量 所以他们在托管堆上的引用地址是一样的。修改age 会对stu1 stu2都造成影响。所以 2种情况下输出结果会不一样。

明白这些 我们就很容易理解参数传递了。

关于参数传递 分按值传递 和按 引用传递

按值传递 分 值类型参数按值快递 和 引用类型参数按值传递

按引用传递分 值类型参数按引用传递 和引用类型参数按引用传递

在讲解参数传递之前 我们还要明白一个概念 形参和实参

 

View Code
1  class Program 2     { 3         static void Main(string[] args) 4         { 5  6             string str = "da jia hao"; 7             //str 是实际参数 8             SayHello(str); 9         }10         //strMessage是 形式参数11         public static void SayHello(string strMessage)12         {13             Console.WriteLine(strMessage);14         }15     }

形参和实参的类型 个数 顺序 必须要完全匹配。

1、值类型参数按值传递:

 

View Code
1        static void Main(string[] args) 2         { 3             int sum = 10; 4             Add(sum); 5             Console.WriteLine(sum); 6         } 7  8         public static void Add(int i) 9         {10             i++;11         }

输出 结果:

我们可以看出sum的值 并没有改变还是10. 我们通过图表来说明为什么会这样。我们知道值类型都是存储在栈上。

2、引用类型参数按值传递:

定义一个引用类型:

View Code
1 public class Student2     {3         public int Age { get; set; }4     }
View Code
1  static void Main(string[] args) 2         { 3             Student stu = new Student { Age =20}; 4             Add(stu); 5             Console.WriteLine(stu.Age); 6         } 7  8         public static void Add(Student stu) 9         {10             stu.Age++;11         }

输出结果:

 

可以看出结果改变了 输出是21.

为什么会是这样呢,直接上图:

 

3、值类型按引用传递

 

View Code
1       static void Main(string[] args) 2         { 3             int sum = 10; 4             Add(ref sum); 5             Console.WriteLine(sum); 6         } 7  8         public static void Add(ref int i) 9         {10             i++;11         }

输出结果:

可以看出 这次sum的 值改变了 不是 10 而是11. 这就是ref 关键字的神奇作用。ref在参数传递中到底起了什么作用呢,我们可以理解为ref 仅仅是一个地址。在文章前面我们提到过, 每个变量都有其堆栈,不同的变量不能共用一个堆栈地址。所以形参和实参在栈上的地址是不一样的。ref所起的作用我们可以简单的理解为他会记住实参的地址,当形参的值改变后,他会把值映射回 之前的实参地址。

4、引用类型按引用传递

View Code
1  public class Student2     {3         public int Age { get; set; }4     }
View Code
1 static void Main(string[] args) 2         { 3             Student stu = new Student { Age=20}; 4             Add(ref stu); 5             Console.WriteLine(stu.Age); 6         } 7  8         public static void Add(ref Student stu) 9         {10             stu.Age++;11         }

 

在使用ref 的时候 调用方法一定要加上ref关键字 ,Add(int i) 和Add(ref int i)是构成方法的重载的。

 

在参数传递中 我们都使用ref 作为示例,那么out关键字 和ref 有什么区别呢。ref和out本质基本一样。唯一的区别就是一个是侧重于修改,一个侧重于输出。

static void Main(string[] args)        {            int sum = 10;            Add(ref sum);            Console.WriteLine(sum);        }        public static void Add(ref int i)        {            i++;        }

这段代码中 我们使用ref 之前 先给 sum 赋了初始值,然后修改sum的值。如果我们使用out 可以先不给sum赋值。在方法内部赋值。

 

params 关键字 是数组类型的参数。可以指定在参数数目可变处采用参数的方法参数,在方法声明中的 params 关键字之后不允许任何其他参数,并且在方法声明中只允许一个 params 关键字。也就是说方法中如果有多个参数只允许有一个params参数,而且要放在方法的最后一个参数。

参见 MSDN示例:

View Code
1 // cs_params.cs 2 using System; 3 public class MyClass  4 { 5  6     public static void UseParams(params int[] list)  7     { 8         for (int i = 0 ; i < list.Length; i++) 9         {10             Console.WriteLine(list[i]);11         }12         Console.WriteLine();13     }14 15     public static void UseParams2(params object[] list) 16     {17         for (int i = 0 ; i < list.Length; i++)18         {19             Console.WriteLine(list[i]);20         }21         Console.WriteLine();22     }23 24     static void Main() 25     {26         UseParams(1, 2, 3);27         UseParams2(1, 'a', "test"); 28 29         // An array of objects can also be passed, as long as30         // the array type matches the method being called.31         int[] myarray = new int[3] {
10,11,12};32 UseParams(myarray);33 }34 }

 

string 类型我们都知道是我们最常用的一种引用类型,那么sting类型的参数传递会是什么样的呢,我们可以自己试验一下,会发现string是很特殊的一个引用类型。后续我会继续讲解关于string类型。 不正之处 欢迎指正!