By value? Or by reference?

原创 2003年03月06日 09:47:00

By value? Or by reference? <?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

传值?还是传引用?

(Wang hailong)

 

关于编程的参数传递问题,总是存在着这样的争论。传值?还是传引用?(还是传指针?还是传地址?)这些提法,经常出现在C++, java, C#的编程技术文档里面。这个问题也经常引起开发人员的争论,徒耗人力物力。实际上,这根本不成为问题,只是由于人为加入的概念,混淆了人们的视听。

从程序运行的角度来看,参数传递,只有传值,从不传递其它的东西。只不过,值的内容有可能是数据,也有可能是一个内存地址

开发人员应该了解程序的编译结果是怎样在计算机中运行的。程序运行的时候,使用的空间可以分为两个部分,栈和堆。栈是指运行栈,局部变量,参数,都分配在栈上。程序运行的时候,新生成的对象,都分配在堆里,堆里分配的对象,栈里的数据参数,或局部变量。

下面举一个C++的例子。

public class Object{

  int i;

  public Object(int i){

       this.i = i;

  }

 

       public int getValue(){

              return i;

       }

 

       public void setValue(int i){

              this.i = i;

       }

};

 

class A {

       Void func1(int a, Object b){

              Object * c = new Object( a );

             

b = c;

       }

 

public      void main(){

       Object * param = new Object( 1 );

 

              func1( 2,  param )

             

              // what is value of parram now ?

              // it is still 1.

    }

};

 

我们来看一下,当调用到func1函数时,运行到Object * c = new Object( a ); 栈和堆的状态。不同编译器生成的代码运行的结果可能会稍有不同。但参数和局部变量的大致排放顺序都是相同的。

<?xml:namespace prefix = v ns = "urn:schemas-microsoft-com:vml" />

运行栈

param

Object

1

return value

main addr

堆空间

main addr

a = 2

b

c

Object

2

func1 addr

<?xml:namespace prefix = w ns = "urn:schemas-microsoft-com:office:word" />

 

这时候,我们来看,param变量被压入运行栈的时候,只是进行了简单的复制。把param里面的内容拷贝到b里面。这时候,b就指向了Object(1)。这里的参数传递,是把param的值传递给b

下面我们来看,程序执行到b = c;时候的堆栈状态。

运行栈

param

Object

1

return value

main addr

堆空间

main addr

a = 2

b

c

Object

2

func1 addr

 

我们可以看到,b现在指向了Object(2)。但是对param的值毫无影响。param的值还是Object(1)

所以,我们说,对参数的赋值不会影响到外层函数的数据,但是,调用参数的操作方法,却等于直接操作外层函数的数据。比如,如果我们在func1()函数中,不调用b=c;而调用b.setValue(3),那么Object(1)的数据就会变为3param的数据也会改变为3

javaC#中的情况,也都是一样。

C++还有一种变量定义方法,表面上看起来,不符合上面的说明,这里进行说明。

Object * a = new Object(1);

Object & * b = a;

这里的b就等于是a的另外一个别名,b就是a。对b赋值就等于对a赋值。甚至作为参数传递时,也是如此。对这种类型的参数的赋值,就等于对外层函数数据的赋值。

public class B{

void func1(Object & * b){

       b = new Object(4);

}

 

public void main(){

       Object * a = new Object(1);

 

       func1(a);

 

       // a is changed to Object(4) now.

}

 

}

当运行完func1(a);时,a的值变化为Object(4)。这是因为编译器实际把参数Object & * b编译为Object ** b_addrb_addr的值是b的地址,也就是a的地址。

当调用func1()的时候,实际上是把b_addr作为参数压到栈里,b_addr的值是a的地址。

当执行b = new Object(4); 时,实际执行了 b_addr->b = new Object(4); 也就是执行了 b_addr->a = new Object(4); a的值当然变化了。

 

还有一点需要说明,当使用COMCORBA等中间件规范进行开发时,我们需要定义IDL语言。参数的类型分为,[in][out][in, out],其中的RPC远程调用的参数打包规范,就更复杂了,但原理却是一样的。

 

关于Swift中的值类型(Value Types)与引用类型(Reference Type)

Swift里面的类型分为两种: ●值类型(Value Types):每个实例都保留了一分独有的数据拷贝。如结构体 (struct)、枚举(enum) 和元组(tuple)。 ●引用类型(Refer...
  • hust_twj
  • hust_twj
  • 2015年08月27日 11:39
  • 1841

swit3.1 (9)笔记 内存管理和方法

Transitioning to ARC Release Notes(oc内存管理) https://developer.apple.com/library/content/releasenot...
  • u014544346
  • u014544346
  • 2017年04月22日 18:16
  • 245

swagger常用注解说明

常用到的注解有: Api ApiModel ApiModelProperty ApiOperation ApiParam ApiResponse ApiResponses ResponseHeade...
  • qq_38765404
  • qq_38765404
  • 2017年12月26日 10:55
  • 56

"undefined reference to" 问题汇总及解决方法 ------非常非常好的一篇文章

转载地址: https://segmentfault.com/a/1190000006049907?utm_source=tuicool&utm_medium=referral 在实际编译代码...
  • stpeace
  • stpeace
  • 2017年06月15日 21:40
  • 1306

文献管理者ReferenceManager开发及使用

经过几天的奋斗,自己终于完成了自己的第一款文献管理工具!!!! 文献管理一直是比较头疼的事情,以前读的文献少的时候,自己还可以在TXT文档中维护好,但是随着文献越来越多,每次添加新的文献都得自己手工...
  • silenceshining
  • silenceshining
  • 2016年08月04日 21:11
  • 1083

JAVA菜鸟入门(12) reference variable是气球的线 +JVM中怎么实现

1 如果variable是primitive,那就拷贝原来变量的值到新变量。 2 如果variable是object referece, 那就拷贝原来reference的值到新的变量,所以就有2个re...
  • feliciafay
  • feliciafay
  • 2015年04月28日 23:42
  • 1650

Java call by value 和 by reference (值传递和引用传递)

关于JAVA中参数传递问题有两种,一种是按值传递(如果是基本类型),另一种是按引用传递(如果是對象).首先以两个例子开始:package com.whf.ByValue_ByReference; /*...
  • chenfengbao
  • chenfengbao
  • 2017年03月24日 09:32
  • 667

dubbo在controller中reference注解为空的问题深度解析

dubbo注解的使用使用非常简单,下面贴出关键部分//provider的配置文件
  • zhou_java_hui
  • zhou_java_hui
  • 2016年11月04日 21:36
  • 11611

C++中引用(reference)的用法详解

C++中引用(reference)的用法详解 TOC 1.简介 2.引用的语法 3.引用使用技巧     3.1 引用和多态     3.2 作为参数     3.3 作为返回值     3....
  • bzhxuexi
  • bzhxuexi
  • 2013年12月26日 18:22
  • 7937

Java基本功之Reference详解

有这样一种说法,如今争锋于IT战场的两大势力,MS一族偏重于底层实现,Java一族偏重于系统架构。说法根据无从考证,但从两大势力各自的社区力量和图书市场已有佳作不难看出,此说法不虚。于是,事情的另一面...
  • StrideBin
  • StrideBin
  • 2017年06月23日 13:43
  • 7248
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:By value? Or by reference?
举报原因:
原因补充:

(最多只允许输入30个字)