深刻理解C#的传值调用和传引用调用

传值调用和传引用调用是几乎所有主流语言都会涉及到的问题,下面我谈谈我对C#中传值调用和传引用调用的理解。

1. 一般对C#中传值调用和传引用调用的理解

  • 如果传递的参数是基元类型(int,float等)或结构体(struct),那么就是传值调用。
  • 如果传递的参数是类(class)那么就是传引用调用。
  • 如果传递的参数前有ref或者out关键字,那么就是传引用调用。

验证示例的代码如下:

view sourceprint?

01    using System;

02

03    public class ArgsByRefOrValue

04    {

05        public static void Main(string[] args)

06        {

07            // 实验1. 传值调用--基元类型

08            int i = 10;

09            Console.WriteLine("before call ChangeByInt: i = " + i.ToString());

10            ChangeByInt(i);

11            Console.WriteLine("after call ChangeByInt: i = " + i.ToString());

12

13            Console.WriteLine("==============================================");

14            // 实验2. 传值调用--结构体

15            Person_val p_val = new Person_val();

16            p_val.name = "old val name";

17            Console.WriteLine("before call ChangeByStruct: p_val.name = " + p_val.name);

18            ChangeByStruct(p_val);

19            Console.WriteLine("after call ChangeByStruct: p_val.name = " + p_val.name);

20

21            Console.WriteLine("==============================================");

22            // 实验3. 传引用调用--类

23            Person_ref p_ref = new Person_ref();

24            p_ref.name = "old ref name";

25            Console.WriteLine("before call ChangeByClass: p_ref.name = " + p_ref.name);

26            ChangeByClass(p_ref);

27            Console.WriteLine("after call ChangeByClass: p_ref.name = " + p_ref.name);

28

29            Console.WriteLine("==============================================");

30            // 实验4. 传引用调用--利用ref

31            Person_ref p = new Person_ref();

32            p.name = "old ref name";

33            Console.WriteLine("before call ChangeByClassRef: p.name = " + p.name);

34            ChangeByClassRef(ref p);

35            Console.WriteLine("after call ChangeByClassRef: p.name = " + p.name);

36

37            Console.ReadKey(true);

38        }

39

40        static void ChangeByInt(int i)

41        {

42            i = i + 10;

43            Console.WriteLine("when calling ChangeByInt: i = " + i.ToString());

44        }

45

46        static void ChangeByStruct(Person_val p_val)

47        {

48            p_val.name = "new val name";

49            Console.WriteLine("when calling ChangeByStruct: p_val.name = " + p_val.name);

50        }

51

52        static void ChangeByClass(Person_ref p_ref)

53        {

54            p_ref.name = "new ref name";

55            Console.WriteLine("when calling ChangeByClass: p_ref.name = " + p_ref.name);

56        }

57

58        static void ChangeByClassRef(ref Person_ref p)

59        {

60            p.name = "new ref name";

61            Console.WriteLine("when calling ChangeByClassRef: p.name = " + p.name);

62        }

63    }

64

65    public struct Person_val

66    {

67        public string name;

68    }

69

70    public class Person_ref

71    {

72        public string name;

73    }

运行结果如下:

看起来似乎上面代码中实验3实验4是一样的,即对于类(class)来说,不管加不加ref或out,都是传引用调用。

其实,这只是表面的现象,只要稍微改一下代码,结果就不一样了。

修改上面代码,再增加两个实验。

001  using System;

002

003  public class ArgsByRefOrValue

004  {

005      public static void Main(string[] args)

006      {

007          // 实验1. 传值调用--基元类型

008          int i = 10;

009          Console.WriteLine("before call ChangeByInt: i = " + i.ToString());

010          ChangeByInt(i);

011          Console.WriteLine("after call ChangeByInt: i = " + i.ToString());

012

013          Console.WriteLine("==============================================");

014          // 实验2. 传值调用--结构体

015          Person_val p_val = new Person_val();

016          p_val.name = "old val name";

017          Console.WriteLine("before call ChangeByStruct: p_val.name = " + p_val.name);

018          ChangeByStruct(p_val);

019          Console.WriteLine("after call ChangeByStruct: p_val.name = " + p_val.name);

020

021          Console.WriteLine("==============================================");

022          // 实验3. 传引用调用--类

023          Person_ref p_ref = new Person_ref();

024          p_ref.name = "old ref name";

025          Console.WriteLine("before call ChangeByClass: p_ref.name = " + p_ref.name);

026          ChangeByClass(p_ref);

027          Console.WriteLine("after call ChangeByClass: p_ref.name = " + p_ref.name);

028

029          Console.WriteLine("==============================================");

030          // 实验4. 传引用调用--利用ref

031          Person_ref p = new Person_ref();

032          p.name = "old ref name";

033          Console.WriteLine("before call ChangeByClassRef: p.name = " + p.name);

034          ChangeByClassRef(ref p);

035          Console.WriteLine("after call ChangeByClassRef: p.name = " + p.name);

036

037          Console.WriteLine("==============================================");

038          // 实验5. 传引用调用--类 在调用的函数重新new一个对象

039          Person_ref p_ref_new = new Person_ref();

040          p_ref_new.name = "old new ref name";

041          Console.WriteLine("before call ChangeByClassNew: p_ref_new.name = " + p_ref_new.name);

042          ChangeByClassNew(p_ref_new);

043          Console.WriteLine("after call ChangeByClassNew: p_ref_new.name = " + p_ref_new.name);

044

045          Console.WriteLine("==============================================");

046          // 实验6. 传引用调用--利用ref 在调用的函数重新new一个对象

047          Person_ref p_new = new Person_ref();

048          p_new.name = "old new ref name";

049          Console.WriteLine("before call ChangeByClassRefNew: p_new.name = " + p_new.name);

050          ChangeByClassRefNew(ref p_new);

051          Console.WriteLine("after call ChangeByClassRefNew: p_new.name = " + p_new.name);

052

053          Console.ReadKey(true);

054      }

055

056      static void ChangeByInt(int i)

057      {

058          i = i + 10;

059          Console.WriteLine("when calling ChangeByInt: i = " + i.ToString());

060      }

061

062      static void ChangeByStruct(Person_val p_val)

063      {

064          p_val.name = "new val name";

065          Console.WriteLine("when calling ChangeByStruct: p_val.name = " + p_val.name);

066      }

067

068      static void ChangeByClass(Person_ref p_ref)

069      {

070          p_ref.name = "new ref name";

071          Console.WriteLine("when calling ChangeByClass: p_ref.name = " + p_ref.name);

072      }

073

074      static void ChangeByClassRef(ref Person_ref p)

075      {

076          p.name = "new ref name";

077          Console.WriteLine("when calling ChangeByClassRef: p.name = " + p.name);

078      }

079

080      static void ChangeByClassNew(Person_ref p_ref_new)

081      {

082          p_ref_new = new Person_ref();

083          p_ref_new.name = "new ref name";

084          Console.WriteLine("when calling ChangeByClassNew: p_ref_new.name = " + p_ref_new.name);

085      }

086

087      static void ChangeByClassRefNew(ref Person_ref p_new)

088      {

089          p_new = new Person_ref();

090          p_new.name = "new ref name";

091          Console.WriteLine("when calling ChangeByClassRefNew: p_new.name = " + p_new.name);

092      }

093  }

094

095  public struct Person_val

096  {

097      public string name;

098  }

099

100  public class Person_ref

101  {

102      public string name;

103  }

则运行结果为:

实验5的运行结果似乎说明即使参数是类(class),只要不加ref,也是传值调用。

下面就引出了我的理解。

2. 没有ref时,即使参数为引用类型(class)时,也可算是一种传值调用

参数为引用类型时,传递的是该引用类型的地址的一份拷贝,“该引用类型的地址的一份拷贝”即为传值调用的“值”。

注意这里说传递的是该引用类型的地址的一份拷贝,而不是引用类型的地址。

下面将用图的形式来说明以上实验3,实验5和实验6中内存的情况。

2.1 首先是实验3

实验3的内存图如下,实参是函数ChangeByClass外的Person_ref对象,形参是函数ChangeByClass内的Person_ref对象。

从图中我们可以看出实参new出来之后就在托管堆上分配了内存,并且在栈上保存了对象的指针。

调用函数ChangeByClass后,由于没有ref参数,所以将栈上的实参p_val拷贝了一份作为形参,注意这里p_val(实参)p_val(形参)是指向托管堆上的同一地址。

所以说没有ref时,即使参数为引用类型(class)时,也可算是一种传值调用,这里的值就是托管堆中对象的地址(0x1000)。

调用函数ChangeByClass后,通过p_val(形参)修改了name属性的值,由于p_val(实参)p_val(形参)是指向托管堆上的同一地址,所以函数外的p_val(实参)的name属性也被修改了。

2.2 然后是实验5

上面的实验3从执行结果来看似乎是传引用调用,因为形参的改变导致了实参的改变。

下面的实验5就可以看出,p_val(形参)p_val(实参)并不是同一个变量,而是p_val(实参)的一个拷贝。

从图中可以看出第一步还是和实验3一样,但是在调用函数ChangeByClassNew后,就不一样了。

函数ChangeByClassNew中,对p_val(形参)重新分配了内存(new操作),使其指向了新的地址(0x1100),如下图:

所以p_val(形参)的name属性改了时候,p_val(实参)的name属性还是没变。

2.3 最后是实验6

我觉得实验6是真正的传引用调用。不废话了,直接上第一个图。

参数中加了ref关键字之后,其实传递的不是托管堆中对象的地址(0x1000),而是栈上p_val(实参)的地址(0x0001)。

所以这里实参和形参都是栈上的同一个东西,没有什么区别了。我觉得这才是真正的传引用调用。

然后调用了函数ChangeByClassRefNew,函数中对p_val(形参)重新分配了内存(new操作),使其指向了新的地址(0x1100)。

由于p_val(形参)就是p_val(实参),所以p_val(形参)的name属性改变后,函数ChangeByClassRefNew外的p_val(实参)的name属性也被改变了。

而原先分配的对象(地址0x1000)其实已经没有被引用了,随时会被GC回收。

3. 结论

  • 如果传递的参数是基元类型(int,float等)或结构体(struct),那么就是传值调用。
  • 如果传递的参数前有ref或者out关键字,那么就是传引用调用。
  • 如果传递的参数是类(class)并且没有ref或out关键字:
    1. 如果调用的函数中对参数重新进行了地址分配(new操作),那么执行结果类似传值调用
    2. 如果调用的函数中没有对参数重新进行了地址分配,直接就是使用了传递的参数,那么执行结果类似传引用调用
时间: 2024-07-31 22:11:14

深刻理解C#的传值调用和传引用调用的相关文章

深刻理解Java中形参与实参,引用与对象的关系

声明:本博客为原创博客,未经允许,不得转载!原文链接为http://blog.csdn.net/bettarwang/article/details/30989755 我们都知道,在Java中,除了基本数据类型之外,其他的都是引用类型,当它们作为函数参数时,传递的也是引用,通过引用可以改变对象的值,很多人便因此而忽略形参与实参,引用与对象的关系问题.废话不多说,先看下面一个例子: import java.util.*; public class Student { private String

Java内存管理-Stackoverflow问答-Java是传值还是传引用?(十一)

勿在流沙筑高台,出来混迟早要还的. 做一个积极的人 编码.改bug.提升自己 我有一个乐园,面向编程,春暖花开! 本文导图: 一.由一个提问引发的思考 在Stack Overflow 看到这样一个问题: Is Java “pass-by-reference” or “pass-by-value”? 翻译成中文: Java是传值还是传引用? 请先不要看下面的内容,思考10秒后,在继续阅读!!! 为什么建议先思考,在阅读内容呢? 我们每天可能会利用碎片化的时间阅读很多内容,有很多信息和知识其实在大脑

Go语言的传参和传引用[转]

目录[-] 传参和传引用的问题 传slice不是传引用! 什么叫传引用? 为什么传slice不是传引用? 为什么很多人误以为slice是传引用呢? 传指针和传引用是等价的吗? 所有类型的函数参数都是传值的! 那Go语言有传引用的说法吗? 什么是引用类型, 和指针有何区别/联系 ? 总结 传参和传引用的问题 很多非官方的文档和教材(包括一些已经出版的图书), 对Go语言的传参和引用的讲解 都有很多问题. 导致众多Go语言新手对Go的函数参数传参有很多误解. 而传参和传引用是编程语言的根本问题, 如

传值和传地址调用

1.原理 传值调用:对形参进行函数变化 传地址调用:对实参进行函数变化 2.实例 #include<iostream>using namespace std;void exchange(int x, int y){             //传入形参    int temp = x;                       //在main函数中x=3,y=4    x = y;    y = temp;    printf("x=%d,y=%d\n", x, y);}

JavaScript系列----数据类型以及传值和传引用

1.简单数据类型 在JavaScript中简单数据类型分为5种.分别为 Undefined, Null,Boolean,Number,String. Undefined类型Undefined类型只有一个值,即特殊的undefined.在使用var对变量声明的时候,变量的值即被初始化为undefined.在使用typeof求得数据类型的时候,对于未声明的变量返回的总是undefined. Null类型Null也只有一个值得数据类型,其实质是一个指向空对象的指针.所以使用typeof操作的时候返回的

python函数传参是传值还是传引用?

首先还是应该科普下函数参数传递机制,传值和传引用是什么意思? 函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题.基本的参数传递机制有两种:值传递和引用传递. 值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本.值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值. 引用传递(pass-

内存(传值、传址、传引用)

前言: 最近对内存产生了兴趣.全是因为马士兵老师的一句Master key.马上要考二级了.大题构造函数比较多.形参有 int & x  这么写的.也有int *x 这么写的.还有普通 int x这么写的.我的乖乖这都是传的什么? int x 是 传值.int *x  是传址. int & x是传引用.那这三者传参数都有什么区别呢? 下面就用三个例子讲解 声明:下面堆栈的图其实是不正确的.八个基本类型(byte.short.int.long.char.float.double.boolea

深刻理解Python中的元类

译注:这是一篇在Stack overflow上很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍然觉得不太明白,希望大家可以给出一些实际的例子和代码片段以帮助理解,以及在什么情况下需要进行元编程.于是e-satis同学给出了神一般的回复,该回复获得了985点的赞同点数,更有人评论说这段回复应该加入到Python的官方文档中去.而e-satis同学本人在Stack Overflow中的声望积分也高达6

深刻理解Python中的元类(metaclass)以及元类实现单例模式

深刻理解Python中的元类(metaclass)以及元类实现单例模式 在看一些框架源代码的过程中碰到很多元类的实例,看起来很吃力很晦涩:在看python cookbook中关于元类创建单例模式的那一节有些疑惑.因此花了几天时间研究下元类这个概念.通过学习元类,我对python的面向对象有了更加深入的了解.这里将一篇写的非常好的文章基本照搬过来吧,这是一篇在Stack overflow上很热的帖子,我看http://blog.jobbole.com/21351/这篇博客对其进行了翻译. 一.理解